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超 文本 预 处 理 器 (Hypertext Preprocessor，PHP) 是 一 种 通用 开源 脚本 语言 ， 其 
语法 吸收 了 C 语言 、Java 和 Perl 的 特点 ， 入 门 门槛 较 低 ， 易 于 学 习 ， 使 用 广泛 ， 
主要 适用 于 Web 开发 领域 PHP 的 文件 扩展 名 为 php。 PHP 独特 的 语法 混合 了 C、 
Java、Perl 以 及 PHP 自 创 的 语法 , 它 可 以 比 CGI 或 者 Perl 更 快速 地 执行 动态 网 页 。 
用 PHP 做 出 的 动态 页 面 与 其 他 编程 语言 相 比 ，PHP 是 将 程序 嵌入 到 HTML( 标 准 
通用 标记 语言 下 的 一 个 应 用 ) 文 档 中 去 执行 ， 执 行 效率 比 完全 生成 HTML 标记 的 
CGI 要 高 许多 。 PHP 还 可 以 执行 编译 后 代码 , 编译 可 以 达到 加 密 和 优化 代码 运行 ， 
使 代码 运行 更 快 的 目的 。 

本 书 是 “工信部 国家 级 计算 机 人 才 评 定 体系 ”中 的 一 本 专业 教材 。“ 工 信和 部 
国家 级 计算 机 人 才 评 定 体系 ”是 由 武汉 厚 溥 教育 科技 有 限 公司 开发 ， 以 培养 符合 
企业 需求 的 软件 工程 师 为 目标 的 IT 职业 教育 体系 。 在 开发 该 体系 之 前 ， 我 们 对 
IT 行业 的 岗位 序列 做 了 充分 的 调研 ， 包 括 研究 从 业 人 员 技 术 方 向 、 项 目 经 验 和 职 
业 素 质 等 方面 的 需求 ， 通 过 对 所 面向 学 生 的 特点 、 行 业 需求 的 现状 以 及 项 目 实施 
等 方面 的 详细 分 析 ， 结 合 我 公司 对 软件 人 才 培 养 模式 的 认 知 ， 按 照 软件 专业 总 体 
定位 要 求 ， 进 行 软件 专业 产品 课程 体系 设计 。 该 体系 集 应 用 软件 知识 和 多 领域 的 
实践 项 目 于 一 体 ， 着 重 培 养 学生 的 熟练 度 、 规 范 性 、 集 成 和 项 目 能 力 ， 从 而 达到 
预定 的 培养 目标 。 

本 书 共 包 括 十 个 单元 : MySQL 概述 、MySQL 基本 操作 、 使 用 SQL 语句 、 高 
级 对 象 、PHP 起 点 、PHP 数组 和 字符 串 、PHP 面向 对 象 、 文 件 上 传 和 异常 处 理 、 
PHP 操作 MySQL， 以 及 Cookie、Session 及 图 像 处 理 。 

我 们 对 本 书 的 编写 体系 做 了 精心 的 设计 ， 按 照 “ 理 论 学 习 一 知识 总 结 一 上 机 
操作 一 课 后 习题 ”这 一 思路 进行 编排 。“ 理 论 学 习 ” 部 分 描述 通过 案例 要 达到 的 
学 习 目 标 与 涉及 的 相关 知识 点 ， 使 学 习 目 标 更 加 明确 ; “知识 总 结 ” 部 分 概括 案 
例 所 涉及 的 知识 点 ， 使 知识 点 完整 系统 地 呈现 ; “上 机 操作 ”部 分 对 案例 进行 了 


使 用 PHP 开发 Web 应 用 程序 


详尽 分 析 ， 通 过 完整 的 步骤 帮助 读者 快速 掌握 该 案例 的 操作 方法 ，“ 课 后 习题 ” 
部 分 帮助 读者 理解 章节 的 知识 点 。 本 书 在 内 容 编 写 方面 ， 力 求 细 致 全 面 ; 在 文字 
叙述 方面 ， 注 意 言 简 意 赎 、 重 点 突出 ; 在 案例 选取 方面 ， 强 调 案例 的 针对 性 和 实 
用 性 。 

本 书 凝聚 了 编者 多 年 来 的 教学 经 验 和 成 果 ， 可 作为 各 类 高 等 院 校 、 高 职高 专 
计算 机 相关 专业 及 培训 机 构 的 教材 ， 也 可 供 广 大 程序 设计 人 员 参 考 。 

本 书 由 武汉 厚 溥 教育 科技 有 限 公 司 编著 ， 由 翁 高 飞 、 余 剑 、 魏 焕 新 、 肖 玉 朝 、 
王 月 、 段 平等 多 名 企业 实战 项 目 经 理 编 写 。 本 书 编者 长 期 从 事项 目 开 发 和 教学 实 
施 ， 并 且 对 当前 高 校 的 教学 情况 非常 熟悉 ， 在 编写 过 程 中 充分 考虑 到 不 同学 生 的 
特点 和 需求 ， 加 强 了 项 目 实战 方面 的 教学 。 本 书 编写 过 程 中 ， 得 到 了 武汉 厚 溥 教 
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Pun 


MySQL 是 MySQL AB 公司 的 数据 库 管理 系统 软件 (2008 年 被 Sun 公司 收购 )， 是 
较 流 行 的 、 开 源 的 关系 型 数据 库 管 理 系统 。 

MySQL 和 Microsoft SQL Server、Oracle 等 软件 一 样 ， 是 一 种 关系 型 数据 库 管 理 系 
统 (Relational Database Management System, 简称 RDBMS), 是 用 于 管理 数据 库 的 软件 系 
统 。MySQL 一 词 中 的 SQL 是 结构 化 查询 语言 (Structure Query Language) 的 缩写 , 是 用 于 
操作 数据 库 的 常用 标准 语言 ， 由 美国 国家 标准 局 (ANSD 和 国际 化 标准 组 织 (ISO) 定 义 。 

MySQL 的 官方 网 站 是 www.mysql.com, 在 这 里 可 以 找到 关于 MySQL 的 各 种 信息 ， 
包括 软件 下 载 、 使 用 手册 、 许 可 说 明 、 应 用 案例 等 。 


1.1 数据 库 基础 


下 面 是 一 些 数据 库 基 本 概念 的 简要 介绍 。 如 果 用 户 已 经 具有 一 定 的 数据 库 基础 和 
应 用 经 验 ， 那 么 以 下 的 内 容 可 以 用 于 学 习 巩 固 ; 如 果 用 户 是 一 个 数据 库 新 手 ， 那 么 以 
下 的 内 容 将 为 学 习 MySQL 提供 必要 的 基础 准备 。 对 于 数据 库 基本 概念 的 理解 也 是 党 


握 MySQL 数据 库 的 一 个 重要 组 成 部 分 ， 如 果 有 必要 ， 可 以 查阅 一 些 有 关 数 据 库 基础 
知识 的 书籍 。 


1.1.1 什么 是 数据 库 


数据 库 是 指 按照 数据 结构 来 组 织 、 存 储 和 管理 数据 的 “仓库 ”。 

在 经 济 管理 的 日 常 工作 中 , 常 需要 把 某 些 相关 的 数据 放 进 这 样 的 “仓库 ”， 并 根据 
管理 的 需要 进行 相应 的 处 理 。 例 如 ， 企 业 或 事业 单位 的 人 事 部 门 常常 要 把 本 单位 职工 
的 基本 情况 (职工 号 、 姓 名 、 年 龄 、 性 别 、 籍 贯 、 工 资 、 简 历 等 ) 存 放 在 表 中 ， 这 张 表 
就 可 以 看 成 是 一 个 数据 库 。 有 了 这 个 “数据 仓库 ”我 们 就 可 以 根据 需要 随时 查询 某 职 
工 的 基本 情况 ， 也 可 以 查询 工资 在 某 个 范围 内 的 职工 人 数 ， 等 等 。 这 些 工作 如 果 都 能 
在 计算 机 上 自动 进行 ， 那 么 我 们 的 人 事 管理 就 可 以 达到 极 高 的 水 平 。 

此 外 ,在 财务 管理 、 仓 库 管 理 、 生 产 管理 中 也 需要 建立 众多 的 这 种 “数据 库 ” 使 
其 可 以 利用 计算 机 实现 财务 、 仓 库 、 生 产 的 自动 化 管理 。 


1.1.2 关系 型 数据 库 
关系 型 数据 库 ， 是 建立 在 关系 模型 基础 上 的 数据 库 ， 借 助 于 集合 代数 等 数学 概念 


和 方法 来 处 理 数据 库 中 的 数据 。 现 实 世 界 中 的 各 种 实体 以 及 实体 之 间 的 各 种 联系 均 用 
关系 模型 来 表示 。 
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关系 型 数据 库 是 目前 最 为 流行 的 大 型 数据 库存 储 与 管理 模式 。 在 关系 型 数据 库 中 ， 
数据 保存 在 一 组 数据 表 中 ， 数 据 表 以 行 、 列 的 形式 存储 数据 ， 类 似 于 电子 表格 。 数 据 
表 中 每 一 行 称 为 一 条 记录 ， 一 条 记录 是 一 组 彼此 相关 的 数据 的 集合 ， 如 一 名 员工 的 姓 
名 、 地 址 、 职 位 等 数据 组 成 的 一 行 记 录 。 每 一 行 记 录 由 一 列 或 者 多 列 组 成 ， 每 一 列 称 
为 一 个 字段 ， 保 存 一 个 单独 的 数据 ， 如 姓名 。 


1.1.3 数据 表 


数据 表 ( 或 称 表 ) 是 数据 库 最 重要 的 组 成 部 分 之 一 。 数 据 库 只 是 一 个 框架 ， 数 据 表 
才 是 其 实质 内 容 。 根据 信息 的 分 类 情况 , 一 个 数据 库 中 可 能 包含 若干 个 数据 表 。 如 “ 教 
学 管理 系统 ”中 , 教学 管理 数据 库 包含 并 围绕 特定 主题 的 6 个 数据 表 :“ 教 师 ” 表 “ 课 
程 ” 表 、“ 成 绩 ” 表 、“ 学 生 ” 表 、“ 班 级 ” 表 和 “授课 ” 表 ， 用 来 管理 教学 过 程 中 学 生 、 
教师 、 课 程 等 信息 。 这 些 各 自 独立 的 数据 表 通 过 建立 关系 被 连接 起 来 ， 成 为 可 以 交叉 
查阅 、 一 目 了 然 的 数据 库 。 

数据 表 是 数据 库 中 一 个 非常 重要 的 对 象 ， 是 其 他 对 象 的 基础 。 没 有 数据 表 ， 关 键 
字 、 主 键 、 索 引 等 也 就 无 从 谈 起 。 

为 减少 数据 输入 错误 ， 并 能 使 数据 库 高 效 工作 ， 表 的 设计 应 按照 一 定 原则 对 信息 
进行 分 类 。 同 时 ， 为 确保 表 结 构 设计 的 合理 性 ， 通 常 还 要 对 表 进 行规 范 化 设计 ， 以 消 
除 表 中 存在 的 元 余 ， 保 证 一 个 表 只 围绕 一 个 主题 ， 并 使 表 容 易 维 护 。 


1.1.4” 列 、 数 据 类 型 和 行 


列 指 表 中 的 一 个 字段 。 所 有 表 都 是 由 一 个 或 多 个 列 组 成 的 。 表 中 的 每 一 列 存储 着 
一 条 特定 的 信息 。 例 如 ， 在 学 生 信 息 表 中 ， 一 个 列 存储 着 学 生 编 号 ， 另 一 个 列 存储 着 
学 生 姓名 ， 而 存储 学 生 家 庭 联系 方式 的 地 址 、 城 市 、 邮 政 编码 全 都 存储 在 单独 的 列 中 。 

如 何 把 需要 存储 的 数据 正确 地 分 解 为 多 个 列 是 极为 重要 的 。 例 如 ， 省 份 、 城 市 、 
邮编 应 该 是 单独 的 列 。 这 样 ， 在 以 后 进行 按 条 件 查 询 、 筛 选 、 排 序 时 ， 将 会 非常 方便 ; 
反之 ， 如 果 这 些 数据 存储 在 一 个 列 ， 以 上 的 操作 将 会 非常 困难 。 

数据 库 中 的 每 一 个 列 都 有 相应 的 数据 类 型 .数据 类 型 定义 列 可 以 存储 的 数据 种 类 。 
例如 ， 学 生年 龄 应 该 用 数字 类 型 ， 家 庭 住址 应 该 用 字符 类 型 ， 入 学 时 间 应 该 用 日 期 时 
间 类 型 ， 等 等 , MySQL 中 可 以 使 用 的 数据 类 型 有 数值 值 、( 字 符 ) 串 值 、 日 期 和 时 间 值 、 
NULL 值 ， 具 体内 容 将 在 后 续 章 节 中 介绍 。 

数据 表 中 的 数据 是 以 行为 单位 存储 的 ， 所 保存 的 每 条 记录 存储 在 自己 的 行内 。 例 
如 ， 在 一 行内 记录 一 个 学 生 的 具体 信息 。 


1.1.5 主键 


主键 是 表 中 数据 行 的 唯一 标识 。 表 中 每 一 行 都 应 该 有 可 以 唯一 标识 自己 的 一 列 (或 
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一 组 列 )。 一 个 学 生 信息 表 可 以 使 用 学 生 编 号 列 作为 唯一 标识 ,商品 信息 表 可 以 使 用 商 
品 编号 作为 唯一 标识 。 

唯一 标识 表 中 每 行 的 这 个 列 (或 这 组 列 ) 称 为 主键 ， 主 键 用 来 表示 一 个 特定 的 行 。 
如 果 没 有 主键 ， 在 对 数据 表 中 的 行进 行 数据 更 新 或 者 删除 的 时 候 ， 就 没有 能 够 找到 唯 
一 数据 行 的 办 法 ， 这 将 给 数据 的 更 新 或 删除 带 来 非常 大 的 安全 隐患 。 也 正 因为 如 此 ， 
虽然 创建 主键 并 不 是 必需 的 ， 但 大 多 数 的 数据 库 设 计 人 员 都 应 该 保证 他 们 创建 的 每 个 
表 都 有 一 个 主键 。 

主键 通常 定义 在 一 个 列 上 ， 但 也 不 是 必需 的 ， 如 果 需 要 ， 可 以 把 多 个 列 一 起 定义 
为 主键 。 无 论 是 单个 列 还 是 多 个 列 作 为 主键 ， 都 应 该 满足 以 下 条 件 。 

e 同一 个 表 中 的 任意 行 的 主键 值 都 不 允许 相同 (不 允许 出 现 重复 的 主键 值 )。 

e 表 中 的 每 一 个 行 都 应 该 具备 主键 值 (不 允许 主键 值 为 NULL)。 

推荐 以 下 几 个 使 用 主键 时 的 好 习惯 。 

。 不 更 新 主键 的 值 。 

。 不 重用 主键 的 值 。 

。 不 把 可 能 会 更 改 的 值 定义 为 主键 。 


1.2 MySQL 简 六 


MySQL 是 一 个 小 型 关系 型 数据 库 管理 系统 ， 开 发 者 为 瑞典 MySQL AB 公司 。 在 
2008 年 1 月 16 号 被 Sun 公司 收购 。 目 前 MySQL 被 广泛 地 应 用 在 Internet 上 的 中 小 型 
网 站 中 。 由 于 其 体积 小 、 速 度 快 、 总 体 拥有 成 本 低 ， 尤 其 是 开放 源码 这 一 特点 ， 使 许 
多 中 小 型 网 站 为 了 降低 网 站 总 体 拥有 成 本 而 选择 了 MySQL 作为 网 站 数据 库 。 


1.2.1 MySQL 的 特性 


MySQL 具有 以 下 主要 特性 。 

使 用 C 和 C++ 编写 ， 并 使 用 了 多 种 编译 器 进行 测试 ， 保 证 源 代码 的 可 移植 性 。 
支持 AIX、FreeBSD、HP-UX、Linux、Mac OS、Novell Netware、OpenBSD、 
OS/2 Wrap、Solaris、Windows 等 多 种 操作 系统 。 

为 多 种 编程 语言 提供 了 API。 这 些 编程 语言 包括 C、C++、Eiffel、Java、Perl、 
PHP、Python、Ruby 和 Tcl 等 。 

e 文 持 多 线程 ， 充 分 利用 CPU 资源 。 

e 优化 的 SQL 查询 算法 ， 有 效 地 提高 查询 速度 。 

。 既 能 够 作为 一 个 单独 的 应 用 程序 应 用 在 客户 端 服务 器 网 络 环 境 中 ， 也 能 够 作 
为 一 个 库 而 嵌入 到 其 他 的 软件 中 提供 多 语言 支持 ， 常 见 的 编码 如 中 文 的 
GB2312、BIG5， 上 日 文 的 Shift_JIS 等 都 可 以 用 作 数 据 表 名 和 数据 列 名 。 
提供 TCP/IP、ODBC 和 JDBC 等 多 种 数据 库 连 接 途 径 。 


单元 一 MySQL 概述 


。 提供 用 于 管理 、 检 查 、 优 化 数据 库 操作 的 管理 工具 。 
。 可 以 处 理 拥有 上 千 万 条 记录 的 大 型 数据 库 。 


1.2.2 MySQL 的 应 用 


与 其 他 大 型 数据 库 如 Oracle、DB2、SQL Server 等 相 比 ，MySQL 自 有 它 的 不 足 之 
处 ， 如 规模 小 、 功 能 有 限 (MySQL Cluster 的 功能 和 效率 都 相对 比较 差 ) 等 ， 但 是 这 丝毫 
没有 减少 它 受 欢迎 的 程度 。 对 于 一 般 的 个 人 使 用 者 和 中 小 型 企业 来 说 ，MySQL 提供 的 
功能 已 经 绰绰有余 ， 而 且 由 于 MySQL 是 开放 源码 软件 ， 因 此 可 以 大 大 降低 总 体 成 本 。 

目前 Internet 上 流行 的 网 站 构架 方式 是 LAMP(Linux+Apache+MySQL+PHP), 即 使 
用 Linux 作为 操作 系统 ，Apache 作为 Web 服务 器 ，MySQL 作为 数据 库 ，PHP 作为 服 
务 器 端 脚本 解释 器 。 由 于 这 四 个 软件 都 是 遵循 GPL 的 开放 源码 软件 ,因此 使 用 这 种 方 
式 可 以 建立 起 一 个 稳定 、 免 费 的 网 站 系统 。 


1.2.3 ”MySQL 的 管理 


可 以 使 用 命令 行 工具 管理 MySQL 数据 库 (命令 mysql 和 mysqladmin)， 也 可 以 从 
MySQL 的 网 站 下 载 图 形 管理 工具 MySQL Administrator 和 MYSQL Query Browser。 

phpMyAdmin 是 由 PHP 写成 的 MySQL 资料 库 系 统管 理 程式 ， 让 管理 者 可 用 Web 
界面 管理 MySQL 资料 库 。 

phpMyBackupPro 也 是 由 PHP 写成 的 ， 可 以 通过 Web 界面 创建 和 管理 数据 库 。 它 
可 以 创建 伪 cronjobs， 可 以 用 来 自动 在 某 个 时 间或 周期 备份 MySQL 数据 库 。 

另外 , 还 有 其 他 的 GUI 管理 工具 ， 如 之 前 的 mysql-front 以 及 ems mysql manager、 


navicat 等 。 
1.2.4 MySQL 许可 说 明 


虽然 MySQL 是 免费 的 开源 软件 ,但 是 MySQL AB 依然 拥有 源 代码 的 版 权 .MySQL 
AB 为 MySQL 制定 了 双重 许可 机 制 : 用 户 可 以 在 遵循 通用 公共 许可 证 的 前 提 下 免费 使 
用 MySQL; 也 可 以 通过 付费 方式 从 MySQL AB 公司 获得 商业 许可 证 。 两 种 许可 方式 获 
得 的 软件 是 相同 的 , 但 许可 证 和 使 用 权限 不 同 。 更 多 信息 参见 http://www .fsf.org/licenses 
上 有 关 GPL 的 详细 内 容 。 

MySQL AB 允许 在 不 发 布 软件 的 前 提 下 ,或 者 在 遵循 GPL 协议 的 前 提 下 使 用 MySQL。 
如 果 你 开发 的 软件 也 遵循 GPL 协议 ， 那 么 可 以 将 MySQL 和 你 的 软件 在 遵循 GPL 的 前 
提 下 一 起 发 布 。 

然而 ， 如 果 你 开发 的 项 目 需要 通过 MySQL 实现 其 功能 ， 并 且 将 依据 收费 许可 证 
出 售 该 软件 ， 则 必须 购买 MySQL AB 的 商业 许可 证 。 还 有 其 他 一 些 情况 可 能 也 会 用 到 
商业 许可 协议 。 关 于 何 时 必须 购买 商业 许可 证 , 请 参见 http://www.mysql.com/company/ 
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legal/licensing 了 解 详 细 信 息 。 


MySQL AB 除了 拥有 MySQL 的 版 权 之 外 ， 还 拥有 其 注册 商标 权 。 因 此 ， 在 你 发 
布 的 软件 中 是 不 允许 出 现 MySQL 的 。 


1.3 ”MySQL 安装 与 配置 


MySQL 支持 多 种 平台 , 在 各 个 平台 下 的 安装 与 配置 过 程 都 比较 简单 , 本 节 将 以 Windows 
环境 下 的 安装 为 例 ， 讲 解 MySQL 的 安装 过 程 。 


1.3.1 MySQL 的 安装 


打 


下 载 的 安装 文件 ， 出 现 如 图 1-1 所 示 的 界面 。MySQL 安装 向 导 启 动 ， 单 


Ff Next 


按钮 继续 。 弹 出 选择 安装 类 型 的 界面 如 图 1-2 所 示 。 


[a 5 
Welcome to the Setup Wizard for MySQL ese 
Server 5.1 Choose the setup type that best suits your needs. 


The Setup Wizard wil nstall MySQL Server 5 1release 5.1.51 
on your computer To continue, cick Next 


WARNING: Ths program sprotectec by copynght aw. (Choose wich program features you wantinstaled and where they 
wl be nstaled, Recommended for advanced users, 


et ee | me | 
图 1-1 


图 1-2 
选择 安装 类 型 , 有 Typical( 默 认 )、Complete( 完 全 )、Custom( 用 户 自 定义 ) 三 个 选项 ， 
我 们 选择 “Custom” 选 项 ， 有 更 多 的 选项 ， 也 方便 熟悉 安装 过 程 。 
选择 安装 内 容 如 图 1-3 所 示 。 


Custom Setwp 
Select the progan featires you want instaled. 


Chdk on an con in the lst below to change how a feature is nstabed. 


Desapton 

旦 :wsQtseve 
Eh 0 ee 

® This fotare vill Ve inatuled on local hard drive, 
8 This featare, snd dl sabfestues, will be installed wn losal bard drive 
全 This featare will he instuled when reguired 


XThis featare will aot be wsilable. 


单元 = 


MySQL 概 述 
在 MySQL Server(MySQL 服务 器 ) 上 单 击 左 键 ， 选择 “This feature, and all 


subfeatures, will be installed on local hard drive.”， 即 “此 部 分 ， 及 下 


属 子 部 分 内 容 ， 全 
部 安装 在 本 地 硬盘 上 ”选项 。 单 击 Change 按钮 ， 手 动 指定 安装 目录 。 
选择 安装 目录 如 图 1-4 所 示 。 


确认 一 下 先前 的 设置 面板 , 如 图 1-5 所 示 , 若 有 误 , 单 击 Back 按钮 返回 重新 设置 。 
若 无 误 ， 单 击 Install 按钮 开始 安装 。 


Change current pestnauon Folder 


Browse to the desiraton fader 


Ready to Install the Program 
mk 本 


| wa severs! 


The wzard 6 ready 加 begn nstalaton 


于 Yeu want to revew ort of ik Bade, Click wm 
Pr bee, honge any of your netalaton metargyr cel 


图 1-4 
正在 安 
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装 的 界面 如 图 1-6 所 示 。 安 装 完成 后 ， 出 现 如 图 1-7 所 示 的 界面 。 


The progran features you selectec are beng nstaled. 


Please wat nhie the Seup Wuard instals MYSQL Server 5.1. Th nay taxe 
Deversl mnutes. 


Saus: 
Capyna new fles 


SS 


图 1-6 


和 击 Next 按钮 继续 ， 出 现 如 图 1-8 所 示 的 界面 。 


ET x 5.1 a Tixard 


Wizard Completed 


Stup has inched nstaingMysQL Server 5.1 Ock Frish to 
st the waard 


MysQ config 
,Setup 3 Whdowes service rurning on a dedicated port 
nd toset the password for the rootaccount. 
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软件 安装 完成 ， 出 现 图 1-8 所 示 的 界面 。 这 里 有 一 个 很 好 的 功能 一 一 MySQL 配置 
向 导 , 不 用 像 以 前 一 样 , 自己 手动 配置 myini 了 , 选中 Configure the MySQL Server now 
前 的 复 选 框 ， 单 击 Finish 按钮 结束 软件 的 安装 并 启动 MySQL 配置 向 导 。 


1.3.2 ”MySQL 的 配置 向 导 


在 图 1-8 中 的 界面 上 单 击 Finish 按钮 ， 出 现 如 图 1-9 所 示 的 界面 ，MySQL Server 
配置 向 导 启 动 。 
单 击 Next 按钮 ， 出 现 如 图 1-10 所 示 的 选择 配置 方式 的 界面 。 


to the MySQL Server Instance er 
ot Configure the MySQL Server 5.1 server instance, 


The Configuration Wizard will alow you to confiour ease select a configuration ype, 
the MySQ. SeverS.1 sener instance, To Contin Rue 
hc Nent. 
Detailed Confiourtion 
| Chooss this configuration type to rreate the optimal sever 
setup for this machine. 


Standard. 
Use ths only on macnines that do not already have 3 MySQL 
Sever nstallaton, This will use 3 general purpose configuration 
Tor the sever that can be tuneg manualy 


me | 上 sm | 
1-9 图 1-10 

MySQL 的 配置 方式 有 Detailed Configuration( 手 动 精确 配置 ) 和 Standard 
Configuration( 标 准 配置 ) 两 种 ， 我 们 选择 Detailed Configuration 配置 方式 ， 方 便 熟 悉 配 
置 过 程 。 

单 击 Next 按钮 ， 出 现 如 图 1-11 所 示 的 选择 服务 器 类 型 界面 。 

MySQL 的 服务 器 类 型 有 Developer Machine( 开 发 测试 类 , MySQL 占用 很 少 资源 )、 
Server Machine( 服 务 器 类 型 , MySQL 占用 较 多 资源 ) Dedicated MySQL Server Machine( 专 
门 的 数据 库 服 务 器 , MySQL 占用 所 有 可 用 资源 ) 三 种 ， 大 家 根据 自己 的 类 型 选择 ， 一 般 选 
择 Server Machine 类 型 ， 不 会 太 少 ， 也 不 会 占 满 。 

单 击 Next 按钮 ， 出 现 如 图 1-12 ee 


Configure the MySQL Server 51 server instance. 


Tea developmen machine, and many ether wl ener purpose dabbases Thiswil optinize the severfor the 
Deronon i WE sener Pouid on usea mlalanount 人 ne ne temo en oathe 


Server Machine 
Several server applications will be running on this machine. 
Choose this option for web/application servers. MySQL will hare 
medium memory usage. 

C Dedicated nysQt Server Mackine 
This machine $s dedicated to run the MySQL Database 


i tet ee a ere 
wil tiize upto al aallable memony. 


| 本 wall 本 Re 
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MySQL 数据 库 的 大 致 用 途 有 Multifunctional Database( 通 用 多 功能 型 ， 好 )、 
Transactional Database Only( 服 务 器 类 型 ， 专 注 于 事务 处 理 ， 一 般 )、Non-Transactional 
Database Only( 非 事务 处 理 型 ， 较 简单 ， 主 要 做 一 些 监控 、 记 数 用 ， 对 MyISAM 数据 类 
型 的 支持 仅 限 于 non-transactional) 三 种 ， 可 根据 自己 的 用 途 来 选择 ， 这 里 选择 
Transactional Database Only 用 途 , 单 击 Next 按钮 继续 ， 出现 如 图 1-13 所 示 的 配置 界面 。 

对 InnoDB Tablespace 进行 配置 ， 即 为 InnoDB 数据 库 文件 选择 一 个 存储 空间 ， 
如 果 修 改 了 ， 要 记 住 位 置 ， 重 装 的 时 候 要 选择 同样 的 地 方 ， 否 则 可 能 会 造成 数据 库 
损坏 ， 当 然 ， 对 数据 库 做 个 备份 就 没 问 题 了 ， 在 此 不 详 述 。 这 里 没有 修改 ， 使 用 默 
认 位 置 ， 直 接 单 击 Next 按钮 继续 ， 出 现 如 图 1-14 所 示 的 界面 。 

[aySQl. Server Instance Configuration Wixa | nstance Configuration Wixar 


er mySQL Server I 


MYSQL Server Instance Configuration MySQL Server Instance Configuration 
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Please select the drive for the InnoDB datafile if you do not want to use the default Please set the approximate number of concurrent connections to the server, 
InnoDB Tablespace Settings Decsion 
Please choose the drve and directory where the innoDE Select thls opllon fol database applications that wlll not requlre 
tablespace should be placed, a high number of concurrent connections, A number of20 
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Volume Name 新 加 着 heavily loaded web servers, 
File System NTFS c 
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选择 用 户 网 站 的 一 般 MySQL 访问 量 ， 同 时 连接 的 数目 有 Decision 
Support(DSS)/OLAP(20 个 左右 )、Online Transaction Processing(OLTP)(500 个 左右 )、 
Manual Setting( 手 动 设置 ， 自 己 输 一 个 数 ) 三 种 ， 可 根据 自己 的 服务 器 进行 选择 ， 这 里 
选择 Online Transaction Processing(OLTP) 选 项 ， 单 击 Next 按钮 继续 ， 出 现 如 图 1-15 
所 示 的 界面 。 
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Please set the server SQL mode, 


5 Enable Strict Mode 


This option forces the serverto behave more like a traditional 
database Seer Ris recomnended to enable this option. 
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图 1-15 
设置 是 否 启用 TCP/IP 连接 ， 并 设 定 端口 。 如 果 不 启用 ， 就 只 能 在 自己 的 机 器 上 访 
问 MySQL 数据 库 ， 这 里 选择 启用 ， 选 中 Enable TCP/IP Networking，Port Number 为 
“3306”， 在 这 个 页 面 上 ， 用 户 还 可 以 选择 “启用 标准 模式 ”(Enable Strict Mode)， 这 样 
MySQL 就 不 会 允许 细小 的 语法 错误 。 如 果 用 户 是 个 新 手 ， 建 议 取消 标准 模式 以 减少 麻 
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烦 。 但 熟悉 MySQL 以 后 ， 尽 量 使 用 标准 模式 ， 因 为 它 可 以 降低 有 害 数据 进入 数据 库 的 
可 能 性 。 还 有 一 个 关于 防火 墙 的 设置 Add firewall exception for this port 需要 选中 ， 但 将 
MySQL 服务 的 监听 端口 加 为 Windows 的 防火 墙 例外 ， 避 免 防 火 墙 阻 断 。 单 击 Next 按 
钮 继续 ， 出 现 如 图 1-16 所 示 的 界面 。 


如 果 要 用 原来 数据 库 的 数据 ， 最 好 能 确定 原来 数据 库 用 的 是 什么 编码 ， 
如 果 这 里 设置 的 编码 和 原来 数据 库 数据 的 编码 不 一 致 ， 在 使 用 的 时 候 可 能 会 
出 现 乱码 。 


MySQL 数据 库 比 较 重 要 的 是 对 MySQL 默认 数据 库 语 言 编码 进行 设置 ， 第 一 个 是 西 
文 编码 , 第 二 个 是 多 字 节 的 通用 UTF8 编码 ， 两 个 都 不 是 我 们 通用 的 编码 ， 人 
第 三 个 ， 然 后 在 Character Set 列 选 框 里 选择 或 输入 gbk， 当 然 也 可 以 用 gb2312， 区 别 就 
是 gbk 的 字库 容量 大 , 包括 了 gb2312 的 所 有 汉字 ,并 且 加 上 了 繁体 字 和 其 他 字 。 使 用 
MySQL 的 时 候 ， 在 执行 数据 操作 命令 之 前 运行 一 次 SET NAMES GBK( 运 行 一 次 就 行 
了 ，gbk 可 以 替换 为 其 他 值 ， 视 这 里 的 设置 而 定 )， 就 可 以 正常 使 用 汉字 (或 其 他 文字 ) 
了 ， 否 则 不 能 正常 显示 汉字 。 单 击 Next 人 出 现 如 图 1-17 所 示 的 界面 。 
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图 1-16 

选择 是 否 将 MySQL 安装 为 Windows 服务 , 还 可 以 指定 Service Name( 服 务 标识 名 
称 )， 是 否 将 MySQL 的 bin 目录 加 入 到 Windows PATH( 加 入 后 ， 就 可 以 直接 使 用 bin 
下 的 文件 ， 而 不 用 指出 目录 名 ， 比 如 连接 mysql.exe -uusername -ppassword 就 可 以 了 ， 
不 用 指出 mysql.exe 的 完整 地 址 ， 很 方便 )， 这 里 选择 了 全 部 ，Service Name 不 变 。 单 
击 Next 按钮 继续 ， 如 图 1-18 所 示 。 

这 一 步 询问 是 否 要 修改 默认 root 用 户 ( 超 级 管理 ) 的 密码 (默认 为 空 )，New root 
password 如 果 要 修改 , 就 在 此 输入 新 密码 (如 果 是 重 装 ， 并 且 之 前 已 经 设置 了 密码 , 在 
这 里 更 改 密码 可 能 会 出 错 ， 请 留 空 ， 并 取消 选中 Modify Security Settings 前 的 复 选 框 ， 
安装 配置 完成 后 另行 修改 密码 )， 在 Confirm( 再 输 一 遍 ) 文 本 框 内 再 输 一 次 ， 防 止 输 错 。 
Enable root access from remote machines( 是 否 人 允许 root 用 户 在 其 他 机 器 上 登录 , 如 果 要 
安全 ， 就 不 要 选择 ， 如 果 要 方 使 ， 就 选择 )。 最 后 建议 可 以 选择 Create An Anonymous 
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Account( 新 建 一 个 匿名 用 户 ， 匿 名 用 户 可 以 连接 数据 库 ， 不 能 操作 数据 ， 包 括 查询 ) 选 
项 ， 设 置 完毕 ， 单 击 Next 按钮 继续 ， 下 


New reot password:  「 Enterthe roct password 
Confim Pe Retype the password. 


克 Enable root access from remote machines 


确认 设置 ， 若 有 误 ， 则 单 击 Back 按钮 返回 检查 。 若 无 误 ， 则 单 击 Execute 按钮 使 
完毕 ， 单 击 Finish 按钮 结束 MySQL 的 安装 与 配置 。 这 里 有 一 个 比较 常见 的 
We service)， 一 般 出 现在 以 前 安装 有 MySQL 的 服务 器 上 。 
解决 的 办 法 是 ， 先 保证 以 前 安装 的 MySQL 服务 器 被 彻底 和 即 载 掉 ， 若 不 行 ， 检 查 是 否 
按 上 面 的 步骤 操作 ， 之 前 的 密码 是 否 有 修改 ; 如 果 依 然 不 行 ， 将 MySQL 安装 目录 下 
的 data 文件 夹 备份 ， 然 后 删除 ， 在 安装 完成 后 ， 将 安装 生成 的 data 文件 夹 删 除 ， 备 份 
的 data 文件 夹 移 回来 ， 再 重启 MySQL 服务 就 可 以 了 ， 这 种 情况 下 ， 可 能 需要 将 数据 
库 检 查 一 下 ， 然 后 修复 一 次 ， 防 止 数据 出 错 。 


【单元 小 结 】 


数据 库 管 理 系统 及 关系 型 数据 库 管理 系统 
数据 库 及 数据 库 中 的 表 、 数 据 行 、 列 、 主 键 
MySQL 的 特性 及 应 用 

MySQL 的 安装 与 配置 


【单元 自 测 】 


1. MySQL 是 (  ) 公 司 的 数据 库 管理 系统 软件 。 

A. Microsoft B. MySQL AB 

C. Sun D. Oracle 
2. MySQL 是 一 种 ( ”) 数 据 库 管 理 系统 。 

A. 层次 型 B. 逻辑 型 C. 关系 型 D. 网 络 型 
3. MYSQL 一 词 中 的 SQL 是 结构 化 查询 语言 ， 由 ( ””) 定 义 。 

A.ANSI 和 ISO B.IEEE 和 WWW 
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C. WWW 和 ANSI D.ISO 和 IEEE 
. 唯一 标识 表 中 每 行 的 这 个 列 称 为 ( )。 

A. 唯一 列 B. 主键 列 C. 标识 列 
. 以 下 描述 中 是 MySQL 特性 的 有 ( ”)。 

A. 为 多 种 编程 语言 提供 了 API 

B. 提供 TCP/IP、ODBC 和 JDBC 等 多 种 数据 库 连接 途径 

C. 支持 多 线程 ， 充 分 利用 CPU 资源 

D. 提供 了 层次 的 数据 库 查 询 方法 ， 实 现 了 层次 的 数据 查询 


MySQL 其 本 棍 作 


时 课程 目标 


PP 了 解 MySQL 中 的 数据 类 型 

PP 了解 MySQL 中 的 函数 

PP 掌握 建立 数据 库 和 数据 表 的 方法 
区 了 解数 据 约束 
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un 


学 习 MYSQL， 需 要 对 数据 类 型 、 函 数 、 数 据 库 、 数 据 表 及 数据 约束 有 深入 了 解 ， 
本 单元 将 会 重点 介绍 MYSQL 数据 类 型 和 函数 ， 以 及 如 何 创 建 数据 库 、 数 据 表 及 数据 
约束 。 


2.1 话 接 MySQL 服务 器 


在 具有 可 供 使 用 的 MySQL DBMS 之 后 ， 对 数据 库 的 各 种 操作 都 可 以 通过 在 
MySQL 命令 行 中 执行 SQL 语句 完成 ， 也 可 以 通过 如 MySQL Query Brower、 
phpMyAdmin 这 种 可 视 化 工具 来 完成 。 本 节 将 重点 介绍 在 MySQL 命令 行 中 执行 SQL 
语句 的 方式 。 


2.1.1 启动 MySQL 服务 


在 “开始 一 控制 面板 一 管理 工具 ”命令 中 找到 “服务 ”选项 ， 双 击 打 开 。MySQL 
服务 的 名 字 在 程序 安装 时 进行 过 设置 ， 默 认 的 服务 的 名 字 是 MySQL。 在 “服务 ”对 
话 框 的 “名 称 ” 列 表 中 找到 它 ， 右 击 ， 在 弹出 的 菜单 中 选择 “启动 ”选项 可 以 启动 
MySQL 数据 库 服务 器 ， 如 图 2-1 所 示 。 


文件 中 操作 查看 VW 帮助 0 


对 已 启动 的 MySQL 数据 库 服务 ， 可 以 右 击 后 选择 “停止 ”按钮 ， 以 停止 MySQL 
数据 库 服务 器 的 运行 。 
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2.1.2 启用 MySQL 命令 行程 序 


在 安装 好 MySQL 并 启动 MySQL 服务 后 ， 有 以 下 两 种 不 同 的 方法 可 启用 MySQL 
的 命令 行 。 

1. 通过 “开始 ”菜单 打开 

在 “开始 ”一 “程序 ”一 MySQL 一 MySQL Server 5.1 一 MySQL Command Line Client 
命令 中 打开 MySQL 的 命令 行 ， 如 图 2-2 所 示 。 

在 Enter password 命令 行 后 输入 root 用 户 的 正确 密码 (root 用 户 密码 在 安装 过 程 中 
配置 )， 出 现 如 图 2-3 所 示 的 欢迎 界面 。 


图 2-2 


2. 使 用 Windows 操作 系统 的 命令 # 


使 用 Windows 操作 系统 的 命令 行 也 可 以 运行 MySQL。 在 Windows 操作 系统 的 命 
令 行 中 输入 mysql 并 按 Enter 键 ， 出 现 如 图 2-4 所 示 的 界面 。 


这 种 方式 是 以 匿名 用 户 的 方法 连接 到 MySQL 数据 库 的 , 在 指令 中 并 未 指定 用 户 名 和 
密码 。 如 果 要 以 特定 的 用 户 名 和 密码 连接 到 MySQL， 请 在 命令 行 中 输入 如 下 指令 。 


mysql -uroot -p 
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按 Enter 键 后 ， 提 示 输 入 该 用 户 的 密码 ， 如 图 2-5 所 示 。 
在 输入 正确 的 密 


码 后 ， 会 出 现 欢迎 界面 ， 如 图 2-6 所 示 。 


图 2-6 


图 2-5 
这 种 是 以 指定 的 用 户 名 和 密码 登录 到 MySQL 数据 库 管 理 系统 ，-u 后 面 紧 跟 用 户 
名 ，-p 后 面 紧 跟 该 用 户 密码 ; 或 者 直接 写 -p， 在 出 现 密码 提示 后 以 星 号 的 显示 方式 输 


入 密码 ;推荐 使 用 后 者 ， 相 对 更 安全 。 


2.1.3 检测 MySQL 命令 行程 序 

在 成 功 开启 MySQL 命令 行程 序 后 ,可 以 在 MySQL 命令 行程 序 中 运行 几 个 简单 的 
SQL 语句 ， 以 测试 一 下 MySQL 命令 行程 序 是 否 正 常 可 用 

查询 当前 登录 的 用 户 名 ， 代 码 如 下 


select current_user(); 


按 Enter 键 后 ， 显 示 的 查询 结果 如 图 2-7 所 示 


图 2-7 


从 查询 结果 可 以 看 到 当前 是 root 用 户 在 localhost 上 登录 到 MySQL 数据 库 服务 


器 的 。 
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单元 二 


; MySQL 中 的 所 有 SQL 指令 都 应 
键 后 该 行 SQL 语句 不 会 执行 ， 而 只 能 


查询 数据 库 系 统 当前 时 间 ， 代 码 如 下 。 


select now(): 


按 Enter 键 后 ， 显 示 的 查询 结果 如 图 2-8 所 示 。 


ysQL nonitor. 
s 18 


图 2-8 


查询 数据 库 双 


中 的 数据 库 ， 代 码 如 下 


show databases; 


按 Enter 键 后 ， 显 示 的 查询 结果 如 图 2-9 所 示 。 


end with 
Connunity 
NO WARRNNTY 。This 
cone to nodify and redistribute it und 
“hv for help. Type 


* to clear the c 


> show database: 


information 


_schema 


MySQL 基 本 操作 


然 ， 检测 命令 行程 序 并 不 是 必需 的 , 但 是 ， 以 上 的 SQL 指令 有 利于 我 们 在 打开 


er ee 服务 器 后 对 当前 的 系统 环境 有 所 了 解 。 
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2.1.4 退出 命令 行 


在 使 用 完 MySQL 命令 行程 序 后 ， 可 以 使 用 Exit 或 Quit 指令 退出 ， 系 统 给 出 Bye 
提示 ， 如 图 2-10 所 示 。 


图 2-10 


2.2 MySQL 的 数据 类 型 


根据 定义 ， 数 据 库 管 a 统 的 目的 就 是 管理 数据 。 即 使 一 条 简单 的 SELECT 1 语 
句 也 涉及 表达 式 求 值 以 产生 一 个 整 型 数据 值 。 

MySQL 中 的 每 个 数据 值 都 有 类 型 。 例 如 ，37.4 是 一 个 数 ， 而 abc 是 一 个 串 。 有 时 ， 
数据 的 类 型 是 明显 的 , 因为 在 使 用 CREATE TABLE 语句 时 指定 了 作为 表 的 组 成 部 分 定义 
的 每 个 列 的 类 型 ， 如 : 


CREATE TABLE my _table 
( 

int_col INT, 

str_col CHAE(20), 

date_col DATE 


) 

而 有 时 ， 数 据 类 型 是 不 明确 的 ， 如 在 一 个 表达 式 中 直接 引用 值 时 ， 将 值 传送 给 

个 函数 ， 或 使 用 从 该 函数 返回 的 值 ， 如 : 

INSERT INTO my _table (int_col,str_col,date_col) values(14,concat("a","b"),19990115) 

用 INSERT 语句 完成 下 列 操作 ， 这 些 操作 全 都 涉及 数据 类 型 。 

e 将 整数 值 14 赋 给 整数 列 int_col。 

e 将 串 值 “a” 和 “b” 传 递 给 函数 CONCATO。CONCAT0O 返 回 串 值 “ab”， 这 
个 串 值 被 赋予 串 列 str_col。 
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e。 将 整数 值 19990115 赋 给 日 期 列 date_col。 而 这 是 不 匹配 的 ， 因 此 ，MySQL 将 
自动 进行 数据 类 型 转换 。 
要 想 有 效 地 利用 MySQL, 必须 理解 它 是 怎样 处 理 数据 的 ,下 面 将 分 别 介 绍 MySQL 
的 数据 类 型 和 列 类 型 。 


2.2.1 MySQL 数据 类 型 


1. 数值 值 

数值 是 诸如 48 或 193.62 这 样 的 值 。 MySQL 支持 说 明 为 整数 (无 小 数 部 分 ) 或 浮 点 
数 (有 小 数 部 分 ) 的 值 。 整 数 可 按 十 进 制 形式 或 十 六 进 制 形式 表示 。 整 数 由 数字 序列 组 
成 。 以 十 六 进 制 形式 表示 的 整数 由 “0x” 后 跟 一 个 或 多 个 十 六 进 制 数字 (“0” 到 “9” 
及 “a” 到 “f”) 组 成 。 例 如 ，0x0a 为 十 进 制 的 10。 浮 点 数 由 一 个 阿拉 伯 数 字 序 列 、 
一 个 小 数 点 和 另 一 个 阿拉 伯 数 字 序列 组 成 。 

2. (字符 ) 串 值 

串 是 诸如 “Madison, SUN ”或 “Tve got a job in the bag.” 这 样 的 值 。 既 可 用 单 引 
号 也 可 用 双 引 号 将 串 值 括 起 来 。 串 中 可 使 用 几 个 转 义 序列 , 它们 用 来 表示 特殊 的 字符 。 
每 个 序列 以 一 个 反 斜 枉 “\” 开 始 ， 表 示 不 同 于 通常 的 字符 解释 。 请 参见 表 2-1 所 示 的 
转 义 序列 表 。 


3. 日 期 和 时 间 值 


日 期 和 时 间 是 一 些 诸如 “1999-06-17” 或 “12:30:43” 这 样 的 值 。MySQL 还 支持 日 
期 /时 间 的 组 合 ， 如 “1999-06-17 12:30:43”。 需 要 特别 注意 的 是 ，MySQL 是 按 “ 年 -月 -日 ” 
的 顺序 表示 日 期 的 。 


4. NULL 值 


NULL 是 一 种 “无 类 型 ”的 值 。 它 表示 的 意思 是 “无 值 ”“ 未 知 值 ”“ 丢 失 的 值 
“溢出 值 ”以 及 “没有 上 述 值 ”等 。 


2.2.2 MySQL 列 类 型 


数据 库 中 的 每 个 表 都 是 由 一 个 或 多 个 列 构成 的 。 在 用 CREATE TABLE 语句 创建 
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一 个 表 时 ， 要 为 每 列 指定 一 个 类 型 。 每 种 列 类 型 都 有 如 下 几 个 特性 。 
其 中 可 以 存放 什么 类 型 的 值 。 
值 要 占据 多 少 空间 ， 以 及 该 值 是 否 是 定 长 的 (所 有 值 占 相同 数量 的 空间 ) 或 可 变 
长 的 (所 占 空 间 量 依赖 于 所 存储 的 值 )。 
该 类 型 的 值 怎样 比较 和 存储 。 
此 类 型 是 否 允许 NULL 值 。 
此 类 型 是 否 可 以 索引 。 
下 面 将 逐一 介绍 常用 列 类 型 的 属性 。 


1. 数值 列 类 型 


MySQL 有 整数 和 浮 点 数值 的 列 类 型 ， 如 表 2-2 所 示 。 
表 2-2 


a 
i 
ee 


FLOAT(4) 最 大 非 零 值 : 

+3.402823466E + 38 FLOAT(4) 4 字 节 
FLOAT(8) 最 大 非 零 值 : FLOAT(8) 8 字 节 
+1.7976931348623157E + 308 


DOUBLE[(M,D)]， 最 小 非 零 值 : 
双 精 度 浮 点 数 +2.2250738585072014E - 308 
一 个 串 浮 点 数 ng 其 值 的 范围 


2. 串 列 类 型 
MySQL 串 列 类 型 如 表 2-3 所 示 。 串 可 以 存放 任何 内 容 ,， 即使 是 如 图 像 或 声音 这 样 
的 绝对 二 进 制 数 据 也 可 以 存放 。 串 在 进行 比较 时 可 以 设 定 是 否 区 分 大 小 写 。 
表 2-3 


定 长 字符 串 CHARCMM 字 节 M 字 节 


VARCHAR 可 变 长 字符 串 VARCHAR(M)M 字 节 
TEXT 小 文本 串 216-1 字 节 
LONGTEXT 大 文本 串 232-1 字 节 


3. 日 期 与 时 间 列 类 型 
MySQL 提供 了 几 种 时 间 值 的 列 类 型 ， 它 们 分 别 是 DATE、DATETIME、TIME、 


单 精度 浮 点 数 
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TIMESTAMP 和 YEAR。 表 2-4 给 出 了 MySQL 为 定义 存储 日 期 和 时 间 值 所 提供 的 这 
些 类 型 ， 并 给 出 了 每 种 类 型 的 合法 取 值 范围 。 


取 值 范围 
日 期 1000-01-01~9999-12-31 
时 间 -838:59:59 一 838:59:59 
1000-01-0100:00:00 一 
日 期 时 间 
9999-12-3123:59:59 


自 1970 年 开始 的 | 19700101000000 一 237 年 的 
时 间 某 个 时 刻 


2.3 建立 数据 库 及 数据 和 表 


在 了 解 了 如 何 使 用 MySQL 命令 行程 序 执行 SQL 语句 及 数据 类 型 后 ， 可 以 尝试 在 
MySQL 中 创建 数据 库 及 数据 表 了 。 


2.3.1 创建 数据 库 


创建 数据 库 应 该 使 用 语句 CREATE DATABASE， 该 语句 的 语法 如 下 。 


CREATE DATABASE [IF NOT EXISTS] db_name 
[create_specification [create_specification] ... ] 


在 以 上 语句 中 ， 各 项 含义 如 下 。 

e CREATE DATABASE 是 关键 字 ， 创 建 数据 库 。 

@ [IF NOT EXISTS] 为 可 选项 ， 当 指定 该 项 时 ， 如 果 已 存在 与 要 创建 的 数据 库 同 
名 的 数据 库 ， 则 不 创建 ， 如 果 不 使 用 该 项 ， 若 已 存在 与 要 创建 的 数据 库 同 名 
的 数据 库 ， 则 报错 。 

e db_name 为 要 创建 的 数据 库 名 称 。 

® create specification 子 句 定 义 创 建 数据 库 时 可 选 的 参数 ， 决 定数 据 库 的 一 些 
属性 。 

注意 ， 在 本 书 以 后 介绍 的 语法 规则 均 采 用 以 下 的 描述 方式 。 

使 用 中 括号 “[]” 括 起 来 的 为 可 选项 ， 可 有 可 无 。 

符号 “|” 表 示 或 ， 用 于 表示 可 选择 项 的 列表 ， 可 在 列表 中 的 选项 是 并 列 的 。 
符号 “{}” 括 起 来 的 ， 表 示 多 个 项 中 必须 选择 一 个 。 

斜体 字 表 示 的 是 由 用 户 定义 的 名 称 或 参数 。 

其 他 为 SQL 关键 字 。 

SQL 中 不 区 分 大 小 写 (但 是 在 Unix 类 操作 系统 中 对 大 小 写 是 有 要 求 的 )。 
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e 各 关键 字 、 名 称 、 参 数 之 间 请 使 用 空格 隔 开 ， 对 于 同 组 内 容 的 各 项 以 “，?” 
分 隔 ( 如 多 个 字段 之 间 可 以 用 去 号 分 隔 )。 
示例 : 为 后 续 章 节 的 示例 创建 数据 库 db test， 代 码 如 下 


mysql>CREATE DATABASE db test; 


以 上 代码 运行 的 效果 如 图 2-11 所 示 。 


Inysql> create database db_tes 
Query OK, 1 row affected (0.86 sec) 


Inysql> show databases; 


4 rows in set 《8.99 sec) 


Inysql> 
图 2-11 


2.3.2 ”删除 数据 库 


删除 数据 库 使 用 DROP DATABASE 语句 ， 该 语句 语法 如 下 
DROP DATABASE db name 


示例 : 删除 当前 数据 


中 名 为 “abc” 的 数据 库 ， 代 码 如 下 。 
mysql>drop database abc 


以 上 代码 运行 的 效果 如 图 2-12 所 示 


5 rows in set (0.88 sec) 


d <@.15 sec》 


Inysql> show databas 
1 Database 


! infornation_schena ! 
1 dhb_test 
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2.3.3 选择 数据 库 


-个 数据 库 管 理 系 统 中 的 数据 库 往往 有 多 个 ， 在 对 数据 库 中 的 任何 对 象 进行 操作 之 
前 ， 首 先 要 选择 数据 库 。 例 如 ， 对 数据 库 中 的 表 、 视 图 、 存 储 过 程 等 进行 查询 、 修 改 、 
删除 等 操作 ， 首 先 要 知道 操作 的 对 象 存放 在 哪个 数据 库 中 ， 选 择 了 对 应 的 数据 库 ， 才 能 
对 该 数据 库 中 的 内 容 进行 操作 。 
选择 数据 库 的 语法 如 下 : 


USE db_ name 
其 中 ，db_name 为 要 选择 的 数据 库 名 ; 例如 ， 在 前 面 创建 的 数据 库 中 选择 db_test 数据 库 
的 代码 如 下 : 
mysql>use db_test; 
可 以 使 用 database() 函 数 检查 当前 MySQL 数据 库 管理 系统 中 选择 的 数据 库 是 哪 
， 以 上 代码 运行 的 结果 如 图 2-13 所 示 


图 2-13 
如 果 没 有 选择 任何 数据 库 ， 则 显示 的 结果 如 图 2-14 所 示 。 


Inysql> t databaseC); 


图 2-14 


2.3.4 创建 数据 表 


创建 数据 表 使 用 CREATE TABLE 语句 ， 其 基本 语法 如 下 。 


CREATE TABLE [IF NOT EXISTS] tbl name 
(create_definition,...) 


[table_option ...] 


e IF NOT EXISTS 表示 当 不 存在 同名 表 时 才 创建 。 
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e tbl name 表示 数据 表 名 称 。 
e create_definition 子 句 定义 表 结构 的 列表 ， 包 括 定义 表 字 段 、 索 引 、 主 键 、 外 
键 等 。 
e table option 定义 表格 属性 的 列表 。 
create_definition 子 句 的 语法 如 下 。 
col name data type [NOT NULL | NULL ] [DEFAULT default value ] 
[ AUTO INCREMENT ] [ UNIQUE [KEY] ] | [PRIMARY KEY] [COMMENT'string'] 


其 中 ，col name 为 字段 名 称 ; data_ type 为 字段 类 型 ，NOT NULL 表示 该 字段 不 允许 
插入 空 值 ，NULL 表示 允许 插入 空 值 ， DEFAULT default value 为 字段 指定 类 认 值 ; 
AUTO_INCREMENT 定义 该 字段 为 自动 增加 ， 默认 情况 下 由 1 开始 增加 ， 当 列 设置 为 
AUTO_INCREMENT 时 ， 需 要 设置 为 NOT NULL; UNIQUE [KEY] 或 者 [PRIMARY] 
KEY 定义 该 字段 为 主键 : [COMMENT'string'] 为 字段 添加 注释 。 

示例 : 某 校 图 书 阅览 室 需 要 一 个 记录 书籍 、 借 阅 人 、 借阅 记录 的 数据 库 。 在 db_test 


裔 要 


数据 库 中 创建 如 表 2-5 所 示 的 数据 表 。 
表 2-5 


CARD 
CNO 


(借阅 人 的 借 书 卡 信息 表 ) 


自 增长 1 


Int 


Varchar 


人 


级 
创建 该 数据 表 的 代码 如 下 : 


mysql> create table CARD( 

-> CNO int(4) primary key auto_increment, 
-> Name varchar(20) not null, 

-> CName varchar(20) not null 


->); 
按 Enter 键 后 ， 创 建 表 成 功 ， 运 行 的 效果 如 图 2-15 所 示 。 


Inysql> create table CARDC 
-> CNO int prinary key auto_increment, 
-> Name varcharC28> not null, 
> CName varchar C20> not null 


» 日 rows affected (0. 


17 sec》 


图 2-15 
如 果 要 查看 当前 数据 库 下 的 表 ， 可 以 使 用 如 下 所 示 的 show tables 语句 。 


mysql> show tables; 
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运行 的 效果 如 图 2-16 所 示 。 


图 2-16 
如 果 要 查看 某 个 表 的 结构 ， 可 以 使 用 SHOW COLUMNS 语句 ， 其 语法 如 下 。 


SHOW COLUMNS FROM tbl name [FROM db name ] 


其 中 ，tbl_name 为 表 名 称 ，db_name 为 数据 库 名 称 。 如 果 要 查看 的 表 在 当前 数据 库 中 ， 
可 以 省 略 数据 库 名 称 ; 如 果 要 查看 的 表 不 在 当前 数据 库 中 ， 除 了 采用 上 述 的 语法 外 ， 
也 可 以 使 用 db_name.tbl name 的 方法 指定 某 个 数据 库 的 某 个 表 。 

示例 : 查看 db_test 数据 库 中 的 CARD 表 结 构 。 代 码 如 下 


mysql> show columns from db_test.card: 


查询 的 结果 如 图 2-17 所 示 。 


图 2-17 


2.3.5 ”删除 表 


删除 表 使 用 DROP TABLE 语句 ， 语 法 如 下 。 
DROP TABLE [IF EXISTS ] tbl_name [,tbl name ] .… 


该 语句 可 以 一 次 删除 一 个 表 或 者 多 个 表 ; IF EXISTS 表示 当 表 存在 时 删除 ， 使 用 
该 参数 可 以 避免 因为 表 不 存在 而 报错 ;tab_name 表示 表 名 。 
示例 : 删除 db_test 数据 库 中 的 card 表 的 代码 如 下 所 示 。 


mysql> drop table card; 


2.3.6” 重 命名 表 


重 命名 表 采 用 RENAME TABLE 语句 ， 其 语法 如 下 。 
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RENAME TABLE tbl name TO new tbl name [ .tbl name2 TO new tbl name2 ] .… 


该 语句 可 以 同时 对 多 个 表 进 行 重 命名 ， 多 个 表 命 名 之 间 以 “, ”逗号 隔 开 。 
示例 : 将 db_test 数据库 中 的 card 表 重 命名 为 stu_card， 代 码 如 下 。 


mysql> rename table card TO stu_card: 


2.4 MySQL 由 的 常用 函数 


与 其 他 大 多 数 计算 机 语言 一 样 , MySQL 支持 利用 函数 来 处 理 数据 。 函数 一 般 是 在 
数据 上 执行 的 ， 它 给 数据 的 转换 和 处 理 提供 了 方便 。 
前 一 部 分 内 容 中 介绍 过 的 用 于 得 到 当前 正 被 用 户 选择 的 数据 库 的 database0) 就 是 一 个 
函数 的 例子 。 
在 MySQL 中， 大 多 数 的 SQL 支持 以 下 类 型 的 函数 。 
用 于 处 理 文本 串 (如 文本 大 小 写 的 转换 、 取 文本 中 的 部 分 字 串 ) 的 文本 函数 。 
用 于 处 理 日 期 和 时 间 值 并 从 这 些 值 中 提取 特定 成 绩 ( 如 返回 两 个 日 期 之 差 、 检 
查 日 期 有 效 性 等 ) 的 日 期 和 时 间 函 数 。 
用 于 在 数值 数据 上 进行 算术 操作 (如 返回 绝对 值 、 进 行 除法 求 余 、 求 平方 根 ) 
的 数学 函数 。 
返回 DBMS 正 使 用 的 特殊 信息 (如 返回 当前 登录 用 户 信息 、 返 回 当前 正 被 选择 
的 数据 信息 ) 的 系统 函数 。 
本 节 将 重点 介绍 前 三 种 ， 系 统 函数 将 在 使 用 到 的 时 候 再 介绍 。 


2.4.1 文本 处 理 函 数 


表 2-6 中 列 出 了 某 些 常用 的 文本 处 理 函数 。 
表 2-6 
函数 

Left() 返回 串 左 边 的 字符 
Length() 返回 串 的 长 度 
Locate() 找 出 一 个 串 中 的 子 串 
Lower() 串 转 为 小 写 
Ltrim() 去 掉 串 左边 的 空格 
RightO 返回 串 右边 的 字符 
Rtrim() 去 掉 串 右边 的 空格 
SubString() 返回 子 串 的 字符 
Upper0 串 转 为 大 写 
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2.4.2 日 期 和 时 间 处 理 函 数 


日 期 和 时 间 采 用 相应 的 数据 类 型 和 特殊 的 格式 存储 。 一 般 ， 应 用 程序 不 使 用 用 来 
存储 日 期 和 时 间 的 格式 ， 因 此 日 期 和 时 间 函 数 总 是 被 用 来 读 取 、 统 计 和 处 理 。 由 于 这 
个 原因 ， 日 期 和 时 间 函 数 在 MySQL 语言 中 具有 重要 的 作用 。 

表 2-7 列 出 了 某 些 常 用 的 日 期 和 时 间 处 理 函 数 。 

表 2-7 

函 数 说 了 明 

AddDate() 增加 一 个 日 期 (天 、 周 等 ) 
AddTime() 增加 一 个 时 间 ( 时 、 分 等 ) 
CurDateO 返回 当前 日 期 
CurTime() 返回 当前 时 间 
Date0 返回 日 期 时 间 的 日 期 部 分 
DateDiff() 返回 两 个 时 间 之 差 
Day0 计算 一 个 日 期 的 天 数 部 分 
Hour0 计算 一 个 日 期 的 小 时 部 分 
Minute() 计算 一 个 日 期 的 分 钟 部 分 
MonthO) 计算 一 个 日 期 的 月 份 部 分 
Now0 返回 当前 日 期 和 时 间 
Second0) 计算 一 个 日 期 的 秒 部 分 
Time() 返回 一 个 日 期 时 间 的 时 间 部 分 
Year() 计算 一 个 日 期 的 年 份 部 分 


2.4.3 ”数值 处 理 函 数 


数值 处 理 函数 仅 处 理 数 值 数 据 ， 这些 函数 一 般 用 于 代数 、 三 角 或 几何 运算 ， 因 此 ， 
没有 串 及 日 期 时 间 处 理 函数 使 用 得 那么 频繁 。 
表 2-8 中 列 出 了 一 些 常 用 的 数值 处 理 函 数 。 


表 2-8 


返回 一 个 数 的 绝对 值 
返回 一 个 角度 的 余弦 
返回 一 个 数 的 指数 值 
返回 除法 的 余数 
返回 圆周 率 
返回 一 个 随机 数 
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返回 一 个 角度 的 正弦 
返回 一 个 数 的 平方 根 
返回 一 个 角度 的 正切 


【单元 小 结 】 


为 执行 SQL 语句 打开 MySQL 的 命令 行程 序 
MySQL 的 数据 类 型 

创建 及 删除 数据 库 和 数据 库 中 的 数据 表 
MySQL 中 的 常用 函数 


【单元 自 测 】 


1. 在 命令 行程 序 中 所 写 的 SQL 语句 应 该 以 ( 。””) 符 号 结束 才能 正确 执行 。 
:i 有 B: 分 号 “3” 
人 D. 什么 符号 都 不 需要 
. 以 下 可 以 正常 退出 命令 行 的 方法 是 ? ( 
A. Exit 回 车 B. Quit 回 车 C.Ctl+C D. Bye 回 车 
. 以 下 关于 数据 类 型 所 占 存 储 空间 描述 不 正确 的 是 ?( ) 
A.INT 占有 4 个 字 节 的 存储 空间 
B. DOUBLE 占有 8 个 字 节 的 存储 空间 
C. DATE 占有 4 个 字 节 的 存储 空间 
D. DATETIME 占有 8 个 字 节 的 存储 空间 
. 以 下 可 以 正常 删除 数据 库 temp 的 SQL 语句 是 ( 。 )。 
A. Drop Database temp: B. Delete Database temp: 
C. Drop Databases temp: D. Delete Databases temp: 
. 以 下 的 SQL 语句 中 ， 能 正确 地 将 stu 表 改 名 为 stuInfo 表 的 是 ( )。 
A. rename table stuInfo TO stu: B. rename table stu TO stulInfo: 
C. rename table stuInfo AS stu: D. rename table stu AS stuInfo: 


【上 机 实战 】 


上 机 目标 


。 简单 操作 MySQL 数据 库 


上 机 练习 


令 第 一 阶段 多 


练习 1: 简单 操作 数据 库 。 


【问题 描述 】 


MySQL 基 本 操作 


练习 打开 MySQL 命令 行程 序 , 练习 创建 数据 库 、 选择 数据 库 、 删 除数 据 库 的 SQL 


语句 。 
【参考 步骤 】 


(1) 按 


照 理论 部 分 所 述 步 骤 打 开 MYSQL 命令 行程 序 ， 直 到 出 现 欢迎 界面 。 
(2) 练习 创建 db_ test 数据 库 ， 代 码 及 运行 效果 如 图 2-18 所 示 。 


图 2-18 


(3) 练习 创建 一 个 以 my_temp 命名 的 临时 数据 库 ， 代 码 如 下 。 


mysql>CREATE DATABASE my temp: 

(4) 删除 上 面 创建 的 my_temp 数据 库 ， 代 码 如 下 。 
Dd op dlabase my emp 

(5) 选择 前 面 创建 的 db_test 数据 库 ， 代 码 如 下 。 


mysql>use db_test 


练习 2: 在 db_test 数据 库 中 创建 借 书 卡 信息 表 。 


【问题 描述 】 


练习 创建 数据 表 ， 练 习 表 字段 数据 类 型 、 主 键 、 非 空 的 不 同情 况 如 何 处 理 。 
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【参考 步骤 】 


(1) 参照 理论 课 步 骤 创 建 表 card。 
(2) 表 的 字段 要 求 如 表 2-9( 与 理论 课 相 同 ) 所 示 。 


表 2-9 
(借阅 人 的 借 书 卡 信息 表 ) 


主键 CNO 
| 字段 名 称 | 字段 说 明 | 类 别 | 位 数 | 属性 | 


可 EE 


| NAME | 迪生 姓 名 | vache | 20 | # 空 | 


(3) 如 果 创 建 过 程 中 部 分 字段 出 错 可 以 把 原 表 删除 再 创建 。 

(4) 尝试 对 表 进 行 重 命名 ， 但 是 要 记得 改 回 card。 

(5) 查阅 其 他 参考 资料 解决 问题 : 如果 CNO 字段 要 求 从 1000 开始 每 次 自 增长 1 
应 该 怎么 办 ? 


练习 3: 在 db_test 数据 库 中 创建 图 书信 息 表 。 


【问题 描述 】 

图 书信 息 表 记 录 了 阅览 室 中 收藏 的 图 书信 息 。 练 习 表 的 创建 、 表 字段 的 处 理 。 
【参考 步骤 】 

(1) 参照 创建 数据 表 的 语法 及 可 用 的 数据 类 型 。 

(2) 图 书信 息 表 的 字段 要 求 如 表 2-10 所 示 。 


表 2-10 


Varchar 非 空 
Varchar 
decimal(9.2) 


Int 


(3) 代码 及 运行 效果 如 图 2-19 所 示 。 


单元 二 ”MySQL 基本 操作 


> create table 


key auto_increnent, 


not null, 


令 第 二 阶段 多 


练习 : 在 db_test 中 创建 图 书 借阅 信息 表 ， 表 字段 要 求 如 表 2-11 
所 示 。 


表 2-11 


表 名 BORROW 
主键 ID 
字段 名 称 字段 说 明 备注 


借阅 编号 


自 增长 1 


外 刍 
BNO 书号 int 外 键 
借阅 日 期 时 间 | dateime| | | 
请 查阅 资料 完成 外 键 的 创建 ， 其 中 CNO 要 作为 外 键 引 用 借阅 卡 信息 表 的 卡号 ， 
BNO 作为 外 键 引 用 书籍 信息 表 的 书籍 编号 ; 而 借阅 日 期 和 时 间 存 储 为 日 期 时 间 类 型 ， 
请 参考 基本 语法 完成 。 


【拓展 作业 】 


1. 创建 一 个 名 为 StuDB 的 数据 库 。 
2. 在 StuDB 数据 库 中 创建 如 表 2-12 所 示 的 班级 信息 表 。 


表 2-12 
表 名 ClsInfo 名 称 班级 信息 表 
主键 | _CID 
序号 字段 名 称 字段 说 明 类 型 位 数 属性 备注 
和 CID 班级 编号 int 4 非 空 主键 自 增长 1 
区 CNAME 班级 名 称 varchar 20 非 空 
3 CreateDate 班级 创建 时 间 datetime 8 非 空 
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3. 在 StuDB 数据 库 中 创建 如 表 2-13 所 示 的 学 生 信息 。 


表 2-13 
实体 名 称 学 生 信息 表 


字段 说 明 属性 备注 
学 生 编号 | i 非 空 自动 编号 ， 主 键 
非 空 


姓名 
年 龄 
了 
所 属 班级 引用 班级 表 班 级 编号 


表 2-14 
SubjectInfo 实体 名 称 科目 信息 表 


SubID 
科目 编号 int 4 非 空 自动 编号 ， 主 键 
eT 
课时 量 int 


5. 在 StuDB 数据 库 中 创建 如 表 2-15 所 示 的 学 生成 绩 表 。 


表 2-15 
GradeInfo 实体 名 称 学 生成 绩 表 
GID 


GID 成 绩 编 号 | int 4 非 空 
科目 编号 4 外 键 引用 科目 表 的 科目 编号 
于 
备注 varchar 200 


使 用 SQL 语 


时 课程 目标 
P 掌握 数据 记录 的 基本 查询 
P 掌握 数据 表 的 增 、 删 、 改 


区 掌握 复杂 数据 查询 
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坦 -7 向 全 


数据 操控 语言 (Data Manipulation Language，DML) 用 于 操作 数据 库 对 象 中 包含 昌 
数据 ， 也 就 是 说 操作 的 单位 是 记录 。 其 主要 包含 增加 、 删 除 、 修 改 和 查询 操作 。 本 单 
元 重点 介绍 数据 基本 查询 、 增 加 、 删 除 、 修 改 及 复杂 查询 。 


3.1 基本 数据 操作 语句 


在 完成 了 数据 库 和 数据 表 的 创建 后 ， 就 可 以 对 数据 表 中 的 数据 进行 操作 ; 这 些 操 
作 包 括 查 询 数据 、 增 加 数据 、 修 改 数据 、 删 除数 据 ， 本 节 将 对 数据 操作 SQL 语句 进行 
详细 介绍 。 


3.1.1 浏览 数据 表 记 录 SELECT 
浏览 数据 记录 使 用 SELECT 语句 ,SELECT 语句 功能 非常 大 , 本 节 只 介绍 SELECT 
语句 的 最 基本 语法 ， 高 级 查询 将 在 后 面 介绍 。SELECT 语句 的 最 基本 语法 如 下 。 
SELECT [*|col name [,col name2,….]] FORM tbl name [WHERE where_condition] 


人 * 表 示 要 查询 的 结果 集中 的 所 有 字段 ; col_ name [col name2,…] 表 示 在 查询 的 结 
果 集 中 选取 部 分 字段 ，FORM 子 句 指定 要 查询 的 表 tbl name; WHERE 子 句 提供 查询 
条 件 ， 将 满足 条 件 的 记录 集 选取 出 来 。 该 语句 将 会 按 指 定 查询 条 件 查询 表 中 所 有 或 部 
分 指定 字段 的 内 容 ， 返 回 满足 条 件 的 数据 记录 。 
示例 ， 对 上 一 章 创 建 的 db_test 数据 库 中 的 card 表 进 行 查询 ， 代 码 如 下 。 


mysql> select * from card; 


以 上 SQL 语句 代码 运行 结果 如 图 3-1 所 示 。 


图 3-1 
查询 结果 将 显示 所 die 表 中 的 数据 记录 ， 如 果 只 需要 数据 表 中 的 部 分 
字段 如 Name 和 CName， 其 SQL 代码 如 下 。 


mysql> select Name,CName from card: 


运行 的 效果 如 图 3-2 所 示 。 
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图 3-2 

以 上 SELECT 语法 中 的 WHERE 子 句 是 带 条 件 的 查询 ,将 在 结果 集中 筛选 出 满足 

条 件 的 结果 以 显示 。WHERE 子 句 的 语法 规则 如 下 。 
exprl [ ,{AND | OR} expr2 [,{AND | OR} expr3 [,.…] ] ] 

其 中 ，AND 表示 与 、 并 且 的 意思 ; OR 表示 或 、 或 者 和 ; AND 和 OR 可 以 嵌 套 使 
用 ， 垦 套 使 用 时 ，AND 级 别人 于 FOR 级 别 ， 也 可 以 通过 “0” 改 变 其 运算 优先 级 。 

示例 : 查询 借阅 卡 信息 表 中 卡号 为 1 的 借阅 人 信息 ， 查 。 上 信息 表 中 卡号 为 
2 的 借阅 人 信息 。 代 码 如 下 


mysql> select * from card where cno=1; 


mysql> select * from card where cno=2; 
根据 前 一 个 示例 的 查询 结果 , 已 经 知道 , 数据 表 中 只 存在 卡号 为 1 的 借阅 人 记录 ， 
不 存在 卡号 为 2 的 借阅 人 记录 ， 因 此 对 卡号 为 2 的 借阅 人 记录 进行 查询 ， 是 没有 结果 
可 以 显示 的 ; 以 上 SQL 运行 的 效果 如 图 3-3 所 示 。 


图 3-3 


3.1.2 插入 数据 INSERT 


向 数据 表 中 插入 记录 使 用 INSERT 语句 ， 其 语法 规则 如 下 


INSERT [INTO] tbl name[(col name, ....)] 
VALUES ({expr | DEFAULT},...),(..),... 


其 中 ，[(col_name,…)] 为 要 插入 值 的 字段 名 称 列表 ; ({expr | DEFAULT},….) 为 插入 值 列 
表 ，expr 为 要 插入 的 值 或 者 表达 式 ，DEFAULT 表示 插入 默认 值 ， 列 和 值 在 语句 中 所 
处 的 位 置 要 一 一 对 应 。 

使 用 INSERT 语句 时 ， 需要 注 意 以 下 几 点 。 

e 字段 名 称 列表 是 可 选 的 ， 但 值 列 表 必须 提供 。 

e 字段 名 称 列表 可 以 是 表 中 的 部 分 字段 ， 也 可 以 是 全 部 字 
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e 有 自 增长 列 时 ， 不 能 为 自 增长 列 提供 值 ， 此 时 必须 填写 字段 名 称 列表 ， 且 字 
段 名 称 列表 中 不 能 包含 自 增长 列 。 

e 为 字段 提供 的 值 和 字段 的 数据 类 型 要 相 匹 配 。 

e 字段 值 要 是 一 个 确定 的 值 、 表 达 式 或 DEFAULT 关键 字 。 

示例 : 在 nd 中 插入 几 条 数据 ， 代 码 如 下 


mysql> insert into card(Name,CName) values(' 四 毛 ",' 自 动 化 2 班 ); 
mysql> insert into card(Name,CName) values(' 五 毛 ',' 电 子 1 班 ); 
mysql> insert into card(Name,CName) values(' 毛 毛 '," 电 子 1 班 ); 
mysql> insert into card(Name,CName) values(' 小 毛 '…… 白 动 化 1 班 ); 


以 上 的 SQL 语句 逐 行 按 Enter 键 ， 逐 行 运行 ， 最 终 插入 到 数据 表 中 的 结果 使 用 
SELECT 语句 查询 ， 如 图 3-4 所 示 。 


图 3-4 


示例 : 为 书籍 信息 表 books 添加 数据 如 下 (后 继 章节 的 案例 将 使 用 此 表 数 据 ): 


insert into books(BName,AUTHOR,Price,Quantity) vlues(' 时 间 这 样 过 ',' 李 曼 曼 ,37.5,400); 
insert into books(BName,AUTHOR.,Price,Quantity) vlues ("每 天 都 ' 同 键 ',56.5,285); 
insert into books(BName,AUTHOR.,Price,Quantity) vlues ("快乐 时 代 ', ' 李 天 琪 ,42.5,185); 
insert into books(BName,AUTHOR.,Price,Quantity) vlues ("每 天 * PHP', ' 李 梦 诗 ,87.5,700); 
insert into books(BName,.AUTHOR.,Price.Quantity) vlues (java 是 这 样 学 的 '' 王 飞 ,32.5,380); 
insert into books(BName,AUTHOR,Price,Quantity) vlues (CHIML 需 懂 得 '", ' 郑 大 ',67.5,200); 
insert into books(BName,AUTHOR,Price,Quantity) vlues (大 学 干什么 ', ' 张 少 龙 ",27.5,300); 
insert into books(BName,AUTHOR,Price,Quantity) vlues (“ 专 业 如 何 选 , ' 赵 六 ',32,200); 
insert into books(BName,AUTHOR,Price,Quantity) vlues (photoshop 学 一 点 ', ' 黎 曼 ',36,450); 
insert into books(BName,AUTHOR,Price,Quantity) vlues (精英 成 长 记 ', ' 李 丽 ',47.5,365); 
insert into books(BName,AUTHOR,Price,Quantity) vlues (幸福 是 什么 ', ' 张 七 虎 ',27.5,300); 
insert into books(BName,AUTHOR,Price,Quantity) vlues ("专业 如 何 选 3', ' 赵 六 ',32,200); 
insert into books(BName,AUTHOR,Price.Quantity) vlues (photoshop 学 精通 ', ' 黎 曼 ',36,450); 
insert into books(BName,AUTHOR.,Price,Quantity) vlues (绘画 技巧 ', ' 李 肖 丽 ,47.5,365); 


对 书籍 表 进 行 查询 ， 结 果 如 图 3-5 所 示 。 
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示例 : 为 书籍 信息 表 books 添加 数据 如 下 (后 继 章 节 的 案例 将 使 用 此 表 数 据 )。 


mysql> insert into borrow(CNO,BNO,RDATE) values(1,1,'2009-9-9"); 

mysql> insert into borrow(CNO,BNO,RDATE) values(2,4,'2009-9-19"); 
mysql> insert into borrow(CNO.BNO.RDATE) values(1,3,'2009-9-19"): 
mysql> insert into borrow(CNO,BNO,RDATE) values(1,3,'2009-9-20"); 


对 借阅 记录 表 进 行 查询 ， 结 果 如 图 3-6 所 示 


RDATE 


2869-89-89 96:89:60 } 
989-89-19 88:99:98 } 
p09 -89-19 98:99:99 } 


889-89-28 99:88:98 } 


图 3-6 


3.1.3 修改 数据 UPDATE 


修改 数据 表 中 的 记录 使 用 UPDATE 语句 , 该 语句 可 以 把 数据 记录 中 的 值 修 改 为 新 值 。 
UPDATE 语句 的 语法 如 下 。 


UPDATE [LOW_PRIORITY] [IGNORE] tbl_ name 
SET col name=exprl] [,col name=expr2...] 
[WHERE where_condition] 

[ORDER BY...] 

[LIMIT row_count] 


其 中 ，LOW_PRIORITY 表示 UPDATE 将 被 延迟 ， 直 到 没有 客户 端 从 表 中 读 取 时 才 
执行 UPDATE 语句 ;IGNORE 表示 即使 在 更 新 过 程 中 出 现 错误 ， 更 新 语句 也 不 会 因 
为 错误 而 中 断 执行 ; SET 表示 要 修改 哪些 值 并 指定 新 值 , 可 以 指定 多 组 “ 列 名 
对 , 其 间 以 “, ”逗号 分 隔 ; WHERE 子 句 表示 对 满足 条 件 的 记录 进行 更 新 ; ORDER BY 
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表示 更 新 将 参照 此 处 所 定义 的 顺序 进行 ， LIMIT 限定 了 更 新 的 行 数 ， 超 出 row_count 


的 行将 会 无 中 止 更 新 。 


示例 : 修改 借阅 人 卡号 为 4 的 记录 ,将 其 所 在 班级 名 称 改 为 “自动 化 2 班 ” 更 新 


的 SQL 代码 如 下 所 示 。 
mysql> update card set CName=' 自 动 化 2 班 ' where CNO=4; 


对 该 数据 记录 进行 更 新 后 ， 对 数据 表 进 行 重新 查询 ， 效 果 如 图 3-7 所 示 。 


Inysql> update card where CNO-4; 
Query OK, 1 row affected (8.87 sec) 
ged: 1 Warnings: @ 


lect * from car 


! Nane 1 CNane 


5 rows in set (9.800 sec) 


Inysql> 


图 3-7 


3.1.4 删除 数据 DELETE 


DELETE 语句 将 表 中 的 数据 记录 删除 ， 其 语法 规则 如 下 


DELETE [LOW_PRIORITY] [IGNORE] FROM tbl name 

[WHERE where_condition] 

[ORDER BY ...] 

[LIMIT row_count] 
语法 规则 中 的 各 项 描述 与 UPDATE 语句 相同 ， . 此 不 再 次 述 。 
示例 : 将 借阅 卡 卡号 为 5 的 借阅 卡 信 息 删 除 ， 其 SQL 代码 如 下 。 


Inysql> delete from card where cno=5: 


代码 运行 的 效果 如 图 3-8 所 示 。 


Inysql> delete from card where cno=5; 
cted (8.04 sec) 
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人 入 着 
在 对 存在 自 增长 列 的 数据 表 进 行 数据 删除 时 ， 已 经 被 删除 的 自 增长 值 将 不 会 


存在 ， 即 使 再 新 增 数据 ， 也 是 在 原来 最 后 一 次 自 增长 值 的 基础 上 增加 。 如 下 示例 ，! 
在 前 面 的 数据 表 上 再 次 增加 “小 毛 ” 这 条 记录 ，CNO 将 为 新 值 6， 而 原来 的 值 5 


将 不 会 存在 ， 如 图 3-9 所 示 。 


DELETE 语句 从 表 中 删除 记录 ; 如果 不 带 WHERE 子 句 ，DELETE 将 从 表 中 删 
除 所 有 的 记录 ， 但 是 DELETE 不 删除 表 本 身 ; 如 果 想 更 快 地 删除 表 中 所 有 的 数据 ， 


可 以 不 使 用 DELETE 语句 而 使 用 TRUNCATE TABLE 语句 ， 它 能 完成 相同 的 工作 ， 
但 是 速度 却 快 得 多 。 实 际 上 ，TRUNCATE 语句 是 将 原来 的 数据 表 删 除 然后 重新 创建 


结构 相同 的 新 表 ， 而 不 是 逐 行 删除 表 中 的 数据 记录 。 
3.1.5 ”更 新 和 删除 的 注意 事项 


在 前 两 部 分 内 容 所 介绍 到 的 UPDATE 和 DELETE 语句 全 部 都 具有 WHERE 子 句 ， 
如 果 没 有 WHERE 子 句 ， 则 UPDATA 和 DELETE 将 会 把 数据 表 中 的 每 一 行 记录 都 发 
生 改 变 或 者 删除 。 这 当然 是 我 们 所 不 期 望 的 。 
以 列 出 了 在 更 新 或 删除 时 应 该 注意 的 事项 。 
e 除非 真 的 打算 更 新 表 中 的 每 一 行 ， 和 否则 ， 请 不 要 使 用 不 带 WHERE 子 句 的 
UPDATE 和 DELETE 语句 。 
e 尽量 保证 每 个 表 都 有 主键 。 这 样 可 以 找到 能 够 唯一 删除 或 更 新 的 条 件 。 
e 在 使 用 UPDATE 或 DELETE 语句 之 前 , 可 以 先 将 WHERE 子 句 放 在 SELECT 
语句 后 面 测试 ， 以 防 编写 的 WHERE 子 句 出 错 。 
e 尽 可 能 实施 引用 完整 性 约束 。 这 样 可 以 避免 被 其 他 表 引 用 的 数据 记录 莫名 其 
妙 地 删除 。 


3.2 高 级 查询 
前 一 部 分 的 介绍 中 , 我 们 已 经 了 解 了 基本 的 SELECT 语句 的 使 用 方法 。 然而, 在 实 
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际 的 应 用 中 ， 对 数据 记录 的 查询 往往 伴随 着 复杂 的 业务 需求 而 存在 。 本 节 将 全 面 介绍 
SELECT 语句 的 高 级 语法 ， 同 时 本 节 还 将 介绍 复杂 的 WHERE 子 句 ， 这 些 WHERE 子 句 
的 使 用 方法 ， 同 样 适用 于 UPDATE 和 DELETE。 在 SQL 查询 语句 的 实现 应 用 中 ， 对 于 
同一 个 业务 需求 ， 能 够 实现 的 SQL 查询 方法 往往 都 有 很 多 种 ， 读 者 可 以 思考 更 多 的 方 
法 实现 ， 而 不 要 拘泥 于 本 书 所 讲 的 示例 。 希 望 本 书 所 介绍 的 SQL 查询 方法 可 以 起 到 抛 
砖 引 玉 的 作用 。 


3.2.1 复杂 查询 
任何 复杂 的 SELECT 语句 ， 其 基本 语法 都 不 会 发 生变 化 ，SELECT 的 语法 参见 上 
1 所 述 。 本 节 将 介绍 几 种 基本 SELECT 语句 上 的 变化 
给 字段 取 别 名 
创建 数据 表 时 ， 数 据 表 的 字段 往往 使 用 英文 、 简 写 或 缩写 ， 往 往 不 易 直 观 获 取 其 
含义 。 为 了 让 查询 结果 直观 易 读 ， 可 以 给 查询 结果 的 字段 取 一 个 直观 的 别名 。 给 字段 
取 别 名 可 使 用 AS 关键 字 。 多 个 字段 之 间 仍 然 使 用 “, ”逗号 分 隔 。 
示例 : 查询 db_test 数据 库 中 的 card 数据 表 ， 代 码 如 下 。 
mysql> select CNO as 卡号 ,Name as 借阅 人 姓名 ,CName as 班级 名 称 from card; 


SQL 语句 运行 结果 如 图 3-10 所 示 。 


图 3-10 
从 运行 结果 中 可 以 看 出 ， 查 询 显 示 的 结果 集中 的 字段 名 已 经 变 成 了 中 文字 段 ， 实 
际 数据 表 中 的 字段 并 没有 发 生 改 变 。 此 举 可 以 让 查询 结果 通俗 易 懂 。 


2. 使 用 表达 式 


在 查询 数据 时 ， 可 以 对 字段 值 进行 运算 处 理 ， 将 处 理 后 的 结果 返回 。 
示例 : 查询 所 有 书籍 信息 ， 并 显示 每 本 书 的 总 成 本 (总 成 本 = 书 价 X 数 量 ); 查询 所 
使 用 的 SQL 语句 如 下 。 


mysql> select BNO,BName,Author,Price*Quantity as 成 本 from books; 


以 上 查询 语句 运行 的 结果 如 图 3-11 所 示 。 
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图 3-11 
在 查询 时 对 字段 进行 的 运算 只 对 显示 的 结果 有 影响 , 对 原 数 据 本 身 没 有 任何 影响 ， 
原 数据 也 不 会 因此 发 生 改变 。 对 数据 进行 运算 可 以 使 用 算术 运算 、 逻 辑 运算 和 比较 运 


算 ; 其 运算 规则 遵循 数学 运算 规则 ， 其 可 以 使 用 的 运算 符 如 表 3-1 所 示 。 


表 3-1 

算术 运算 符 +( 加 )、-( 减 )、*( 乘 )、/( 除 )、DIV( 除 )、%( 余 )、MOD( 余 ) 

<( 小 于 )、>( 大 于 )、<( 小 于 等 于 )、 盖 (大 于 等 于 )、<>( 不 等 于 )、!=( 不 等 于 )、 
=( 等 于 ) 

逻辑 运算 符 NOT( 非 )、! ( 非 )、AND( 并 且 )、&&( 并 且 )、OR( 或 )、||( 或 )、XOR( 异 或 ) 

IS NULL( 为 空 )、IS NOT NULL( 不 为 空 )、 BETWEEN m Andn( 在 m 到 n 之 间 ， 
大 于 等 于 m 且 小 于 等 于 n) 


3. 对 查询 结果 进行 排序 


在 SELECT 的 语法 中 有 一 个 ORDER BY 子 句 ， 该 子 句 可 以 按 所 指定 的 条 件 对 查 
询 结 果 集 进行 排序 ， 该 子 句 的 语法 规则 如 下 : 


比较 运算 符 


其 他 


ORDER BY col name [ desclasc ] [.col name2 [desclasc] ...] 


其 中 ，col_name 是 排序 Ss 字段 名 列表 ， 字 段 名 列表 中 有 多 个 字段 时 ， 依 据 字段 名 
列表 顺序 ， 首 先 按 第 一 个 字段 排序 ， 在 相同 的 排序 结果 上 按 第 二 个 字段 排序 ， 并 依次 
类 推 ，desc ee asc 指 从 低 到 高 排 ，desclasc 可 以 省 略 不 写 ， 在 省 略 的 情况 下 
默认 按照 asc( 也 就 是 从 低 到 高 ) 的 方式 排序 。 

示例 : 查询 书籍 信息 表 ， 按 书 价 从 低 到 高 排序 ， 查 询 的 SQL 语句 如 下 。 


mysql> select * from books order by price; 


运行 结果 如 图 3-12 所 示 。 
示例 : 查询 书籍 信息 表 ， 按 书籍 现存 数量 从 高 到 低 排序 ， 查 询 的 SQL 语句 如 下 。 


mysql> select * 位 om books order by quantity desc; 


以 上 SQL 语句 运行 的 结果 如 图 3-13 所 示 。 
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专业 如 何 选 3 
专业 如 何 
快乐 时 代 


4. 限制 返回 结果 的 行 数 


通过 SELECT 语句 的 LIMIT 子 句 可 以 限制 返回 结果 的 行 数 。 例 如 ， 将 书籍 信息 按 序 
号 从 高 到 低 排序 后 ， 查 询 书 籍 信息 表 中 第 6 本 书 开 始 的 四 本 书 。 其 SQL 语句 如 下 所 示 。 


mysql> select * from books order by bno desc LIMIT 5.4; 


运行 的 结果 如 图 3-14 所 示 。 


| 大 学 干什么 
| HTIL 需 懂得 
n set (0.00 
图 3-14 
其 中 ，LIMIT 子 句 后 指定 了 两 个 整数 参数 ， 第 一 个 参数 表示 开始 行 的 偏 移 量 (0 表示 第 


- 行 )， 第 二 个 参数 表示 要 返回 的 记录 的 条 数 。 当 LIMIT 子 句 的 后 面具 写 一 个 参数 N 
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时 ， 表 示 从 表 第 一 条 记录 开始 取 NN 条 记录 
5. 消除 重复 的 行 


使 DISTRCY le 结果 中 完全 相同 的 行 ， i \ 返 回 一 条 。 
例如 ， 查询 所 有 的 学 生 班 级 名 称 ， 相 同 的 班级 名 称 只 显示 一 个 。 查 询 的 SQL 语句 如 下 
所 示 : 


mysql> select DISTINCT CName from card; 


该 SQL 语句 执行 的 结果 如 图 3-15 所 示 。 


DISTINCT CNane from card; 


图 3-15 
DISTINCT 关键 字 后 字段 列表 可 以 是 一 个 字段 或 是 多 个 字段 ， 当 多 个 字段 时 , 只 有 当 
多 个 字段 内 容 均 相同 时 ， 两 行 + 被 认为 是 重复 的 


3.2.2 ”模糊 查询 


前 面 使 用 的 所 有 查询 ， 其 共同 点 是 针对 已 知 值 进行 匹配 。 不 管 是 匹配 一 个 还 是 多 
个 值 ， 测 试 大 于 还 是 小 于 已 知 值 ， 或 者 检查 某 个 范围 的 值 ， 共 同 点 是 条 件 中 使 用 的 值 
都 是 已 知 的 。 在 现实 应 用 中 , 很 多 查询 都 是 只 知道 部 分 条 件 , 例如 , 要 查询 所 有 姓 “ 王 ” 
的 借阅 人 的 借阅 卡 信息 。 这 就 需要 使 用 到 模糊 查询 。 

模糊 查询 使 用 LIKE 关键 字 进 行 ， 配 合 LIKE 完成 工作 的 ， 还 有 两 个 通配符 “% 
和 “_”。 其 中 ,“%” 表 示 在 出 现 它 的 位 置 匹 配 任意 多 个 字符 ,“_” 表 示 在 出 现 它 的 位 
置 匹配 一 个 字符 。 

示例 : 在 书籍 信息 表 中 查找 所 有 包含 “java” 字 符 的 书籍 信息 ， 查 询 的 SQL 语句 如 
下 所 示 。 


mysql> select * 位 om books where bname like '%javao%0': 


该 SQL 语句 执行 的 结果 如 图 3-16 所 示 。 
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使 用 模糊 匹配 需要 注意 如 下 两 个 问题 。 

(1) 串 尾 的 空 串 尾 的 空格 因为 看 不 见 有 可 能 会 被 忽视 ， 如 “Hello World”( 尾 
部 有 一 空格 ) 如 果 用 “%World” 来 匹配 是 不 成 功 的 ， 而 该 字 串 尾部 的 空格 却 被 不 经 意 
间 忽 视 了 。 

(2) NULL。 似 乎 “%” 通 配 符 可 以 用 来 匹配 任何 东西 ， 但 有 一 个 例外 ， 那 就 是 
NULL。 换 句 话说 “LIKE %” 能 匹配 NULL 值 。 


3.2.3” 子 查询 


了 查询 是 指 在 一 条 SQL 语句 中 嵌入 另 一 个 SELECT 语句 的 查询 。 
1. 使 用 IN 和 NOT IN 的 子 查询 
IN 操作 符 是 用 来 指定 条 件 范围 的 ， 范 围 中 的 每 个 条 件 都 可 以 进行 匹配 。NOT IN 表示 
不 在 条 件 范围 中 。 例 如 : 查询 有 借阅 记录 的 借阅 卡 信息 ， 可 以 使 用 的 SQL 语句 如 下 所 示 。 
mysql> select * from card where CNO in(select CNO from borrow); 
在 以 上 的 查询 语 名 中， 查询 的 条 件 需 要 匹配 另 一 个 查询 的 结果 ， 这 是 一 种 典型 的 
了 查询。 运行 效果 如 图 3-17 所 示 


图 3-17 
如 果 WHERE 子 句 使 用 了 大 于 、 小 于 、 大 于 等 于 、 等 于 之 类 的 比较 运算 符 


查询 能 得 到 合适 的 匹配 值 时 ， 也 可 以 在 运 
2. 使 用 EXISTS 和 NOT EXISTS 的 子 查询 


算 符 后 面 加 上 子 查询 。 读 者 可 以 自行 实验 。 


EXISTS 关键 字 是 用 于 判断 查询 是 否 存在 的 关键 字 ， 如 果 查 询 存 在 则 返回 值 为 真 ， 
否则 返回 值 为 假 。 例 如 ， 如 果 存 在 借阅 卡 卡号 为 2 的 借阅 记录 ， 就 将 该 借阅 卡号 所 对 
应 的 借阅 卡 信息 查询 出 来 ， 对 应 SQL 语句 代码 如 下 。 


mysql> select * from card where cno=2 and exists (select 1 位 om borrow where cno=2); 


以 上 代码 运行 的 效果 如 图 3-18 所 示 。 


图 3-18 
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其 中 ，select 1 起 到 的 作用 只 是 在 有 满足 条 件 的 情况 下 查询 出 一 个 常量 1， 如 果 存 在 党 
量 1 就 查询 出 结果 ， 如 果 不 存 量 1 就 不 进行 查询 。 


3.2.4 聚合 函数 


实际 应 用 中 往往 需要 对 表 中 的 数据 (而 不 是 实际 数据 本 身 ) 进 行 汇总 ， 为 此 MySQL 
提供 了 专门 的 函数 ， 这 就 是 聚合 函数 。 
表 3-2 列 出 了 常用 的 几 个 聚合 函数 。 


表 3-2 
函数 说 明 
AVGO | 返回 某 列 的 平均 值 
COUNTO | 返回 某 列 的 行 数 
MAX0 返回 某 列 的 最 大 值 
MINO 返回 某 列 的 最 小 值 
SUM 返回 某 列 值 的 和 


示例 : 查询 阅览 宝藏 书 的 总 数量 ， 查 询 的 SQL 语句 如 下 。 


mysql> select sum(quantity) as ' 藏 书 总 量 ' from books; 


以 上 代码 运行 的 结果 如 图 3-19 所 示 


图 3-19 


示例 : 查询 阅览 室 藏 书 的 平均 价格 ， 查 询 的 SQL 语句 如 下 。 
mysql> select avg(price) as 平均 书 价 ' from books; 


以 上 代码 运行 的 结果 如 图 3-20 


avgCprice) as 


所 i 


图 3-20 


有 时 我 们 需要 对 表 中 的 数据 进行 分 组 ， 对 每 一 个 分 组 数据 进行 再 统计 ， 这 就 需要 
使 用 到 GROUP BY 语句 。 例 如 ,查询 有 借阅 记录 的 每 一 个 借阅 卡号 分 别 借 了 多 少 本 书 。 
其 实现 该 功能 的 SQL 语句 如 下 。 


mysql> select ID,count(ID) as 借阅 书籍 总 数 from borrow group by CNO; 
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以 上 代码 运行 的 效果 如 图 3-21 所 示 。 


Inysql> select 1D,countC1D》as 借 数 fron borrow group by CNO; 


in set (8.80 sec) 


如 果 需 要 对 分 组 聚合 的 结果 进行 按 条 件 再 查询 ， 就 需要 使 用 到 配合 GROUP BY 使 用 
的 HAVING 子 句 。 HAVING 子 句 和 WHERE 子 句 的 作用 相同 , 但 是 它 一 定 要 配合 GROUP 
BY， 并 对 分 组 聚合 的 结果 进行 再 过 滤 时 使 用 。 
示例 : 查询 有 借阅 记录 的 每 一 个 借阅 卡号 分 别 借 了 多 少 本 书 ， 把 借阅 书籍 总 数 在 
2 本 以 上 的 信息 显示 出 来 。 其 代码 如 下 。 
mysql> select ID,count(ID) as 借阅 书籍 总 数 from borrow group by CNO HAVING 
count(ID)>2; 


代码 运行 的 效果 如 图 3-22 所 示 。 


数 from borrow group by CNO HRUING count 


图 3-22 


3.2.5 ”多 表 联 合 
前 面 介绍 的 查询 方法 都 是 针对 单 表 进行 ， 实 际 的 应 用 中 经 常会 出 现 需 要 从 几 个 表 
中 同时 取 数 据 的 情况 ， 这 就 需要 使 用 到 多 表 联 合 查询 。 以 下 介绍 几 种 多 表 联 合 查询 。 
1. 使 用 FROM 子 句 的 多 表 联 合 


最 简单 的 多 表 联 合 查询 是 在 FROM 子 句 部 分 指定 多 个 表 名 ， 每 个 表 名 之 间 以 “,” 
了 分 隔 。 代 码 如 下 所 示 。 
Select * from tbl _ namel,tbl_ name2 

此 时 ， 如 果 tbl namel 中 有 记录 5 条 ，tbl name2 中 有 记录 6 条 ， 此 举 将 产生 5X6=30 条 
记录 ; 换 句 话说 ，FROM 中 的 多 表 联 合 查询 实际 上 是 把 第 一 个 表 中 的 每 一 条 记录 和 第 二 
个 表 中 的 每 一 条 记录 相 匹 配 ， 也 就 是 数学 中 讲 的 “第 卡尔 积 ” 这 种 数据 查询 方法 并 不 实 
用 ， 试想， 如 果 把 全 国 户籍 信息 系统 的 人 口 信息 表 和 人 口 所 属 的 省 份 信息 表 进行 联合 查 
询 ， 若 使 用 这 种 方法 ， 产 生 的 记录 数量 娩 怕 不 是 随便 哪个 服务 器 能 够 承担 的 ， 如 果 再 联 
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合 街道 信息 表 ， 估 计 服 务 器 就 会 因为 这 条 查询 语句 而 死机 。 
co -种 好 的 多 表 联 合 查询 方法 ， 在 此 不 作 示 例 ， 也 不 推荐 
使 用 这 种 多 表 联 合 查询 ， 有 兴趣 的 可 以 自行 实验 。 


2. 使 用 内 连接 (INNER JOIN) 


内 连接 是 最 常用 的 一 种 多 表 联 合 方式 ， 通 过 两 个 表 中 具有 的 共同 值 将 两 个 表 中 的 记录 
连接 在 一 起 。 
示例 : 查询 有 借阅 记录 的 用 户 信息 及 借阅 记录 信息 ， 代 码 如 下 。 


mysql> select * from card inner join borrow on card.CNO=borrow.CNO; 
card.CNO=borro 


! BNO ! RDATE 


9-99-99 99:99:99 } 


9-89-19 99:99:99 } 


1 
4 9 -8 99:B9:98 ! 
3 

3 9-89-28 99:698:98 } 


set (0.06 sec) 


图 3-23 


3. 使 用 外 连接 


外 连接 的 多 表 联 合 查询 又 分 为 左 外 连接 、 右 外 连接 、 交 叉 连 接 三 种 不 同 的 方式 。 
e 左 外 连接 (LEFT JOIN): 以 左 表 为 主 ， 保 持 左 表 的 数据 不 变 ， 右 表 的 数据 与 之 


匹配 。 
e 右 处 连接 (RIGHT JOIN): 以 右 表 为 主 ， 保 持 右 表 的 数据 不 变 ， 左 表 的 数据 与 
之 匹配 。 


e 交叉 连接 (CROSS JOIN): 与 内 连接 的 效果 完全 相同 ， 在 此 不 再 袭 述 。 
示例 : 查询 所 有 的 借阅 卡 信 息 ， 如 果 有 借阅 记录 的 ， 同 时 显示 其 借阅 记录 ， 实 现 
该 查询 的 SQL 语句 如 下 。 


mysql> select * 位 om card left join borrow on card.CNO=borrow.CNO:; 


以 上 代码 运行 效果 如 图 3-24 所 示 。 


99-89-28 98:99:89 
09-99-19 99:99:99 
LL 
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【单元 小 结 】 


对 数据 表 中 的 数据 进行 基本 的 增加 、 删 除 、 修 改 、 查 询 操作 
复杂 查询 及 子 查 询 的 使 用 

聚合 函数 、 分 组 聚合 及 带 条 件 的 分 组 聚合 的 使 用 
常用 的 多 表 联 合 查 询 


元 自 测 】 


下 


以 下 代码 中 不 能 删除 表 tbl_temp 中 所 有 数据 的 是 (  )。 
A. Delete from tbl_ temp: B. Delete from tbl _ temp where 1=1: 
C. Delete from tbl temp having 1=1: D.TRUNCATE tbl temp: 


- 以 下 代码 中 不 能 正常 运行 的 是 (。”)。 


A. Select col namel as A,col name2 as B from tb]l name; 
B. Select col namel A,col name2 B from tb]l name:; 

C. Select A=col namel,B=col name from tbl name; 

D. Select col namel,col name2 from tb]l name:; 


. 关于 分 组 聚合 查询 ， 以 下 说 法 不 正确 的 是 (。”)。 


A. 查询 语句 的 字段 列表 中 必须 包含 至 少 一 个 聚合 函数 

B. 查询 语句 的 字段 列表 中 可 以 没有 聚合 函数 

C. 查询 语句 的 字段 列表 没有 包含 在 聚合 函数 中 的 列 必 须 包含 在 GROUP BY 子 
句 中 

D. GROUP BY 子 句 中 可 以 按照 一 列 或 者 多 列 进行 分 组 


. 以 下 代码 中 ， 能 按 ID 字段 升序 显示 的 选项 有 ( ”)。 


A. Select * from tbl name order ID 

B. Select * from tbl _ name order ID desc 

C. Select * 位 om tbl name order by ID asc 
D. Select * from tbl name order by ID desc 


. 查询 包含 “入 门 ” 字 串 作 为 书 名 的 查询 语句 的 LIKE 子 句 有 ( 。 ”)。 


A. LIKE “% 入 门 ” B. LIKE “% 入 门 %” 
C LIKE NI DE 关门” 


【上 机 实战 】 


上 机 目标 


。 熟练 掌握 在 数据 表 中 增 、 删 、 改 、 查 数据 的 SQL 语句 
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掌握 如 何 对 数据 表 进 行 复杂 查询 
掌握 模糊 查询 、 子 查询 、 多 表 联 合 查询 的 查询 方法 
e 了 解 并 练习 分 组 聚合 技术 


未 


上 机 练习 


令 第 一 阶段 


练习 1: 为 StuDB 数据 库 中 的 数据 表 添 加 测试 数据 。 


【问题 描述 】 


二 全 抽 生 全 为 学 生 信息 表 
StuInfo 添加 至 少 五 条 测试 数据 , 为 科目 信息 表 SubjectInfo 添加 至 少 三 条 测试 数据 ,为 
学 生成 绩 表 GradeInfo 添加 至 少 五 条 测试 数据 。 在 完成 此 操作 后 可 以 尝试 对 数据 表 中 
的 数据 进行 修改 和 删除 操作 。 


【问题 分 析 】 
本 练习 主要 是 学 习 如 何 操 作 数据 表 中 的 数据 . 
【参考 步骤 】 


(1) 班级 信息 表 ClsInfo 添加 的 测试 数据 如 下 所 示 。 


insert into clsinfo(cname,createdate) values(' 高 三 1 班 ''"2009-8-25)): 
insert into clsinfo(cname,createdate) values(' 遍 三 2 班 '"2009-8-25"); 


insert into clsinfo(cname,createdate) values(' 遍 三 3 班 ''2009-8-26"); 


添加 后 的 表 数 据 如 图 3-25 所 示 。 


Inysql> select * from clsinfo; 


2889-88-25 99:99:99 ! 


2989-98-25 99:99:98 ! 


图 3-25 
(2) 为 学 生 信息 表 StuInfo 添加 至 少 五 条 测试 数据 。 


insert into stuinfo(name,age,sex,cid) values('sanmao',18,'F',1); 


insert into stuinfo(name,age,sex,cid) values(simao',1 


insert into stuinfo(name,age,sex,cid) values(‘wumao',19,'F',2); 
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insert into stuinfo(name,age,sex,cid) values(‘wumao',20,M',1); 
insert into stuinfo(name,age,sex,cid) values('maomao',18,M',1); 
insert into stuinfo(name,age,sex,cid) values('xiaomao',19,M',2); 


(3) 为 科目 信息 表 SubjectInfo 添加 至 少 三 条 测试 数据 。 


insert into SubjectInfo(SubName,SubHour) values("Math',120); 
insert into SubjectInfo(SubName,SubHour) values('Chinese',108); 
insert into SubjectInfo(SubName,SubHour) values(‘English',98); 


(4) 为 学 生成 绩 表 GradeInfo 添加 至 少 五 条 测试 数据 。 


insert into GradeInfo(SID,SubID,Grade) values(1,1,99); 
insert into GradeInfo(SID,SubID,Grade) values(1,2,79): 
insert into GradeInfo(SID.,SubID,Grade) values(1.3,89); 
insert into GradeInfo(SID,SubID,Grade) values(2.1.96); 
insert into GradeInfo(SID,SubID,Grade) values(2.2,69); 
insert into GradeInfo(SID,SubID,Grade) values(2.3,75); 
insert into GradeInfo(SID,SubID,Grade) values(3,1,55); 
insert into GradeInfo(SID,SubID,Grade,Remark) values(3,2.NULL,'ABSENCE); 


练习 2: 查询 所 有 没有 考试 成 绩 记录 的 学 生 信息 。 


【问题 描述 】 


成 绩 记 录 表 中 的 Grade 字段 记录 了 学 生成 绩 ， 所 有 成 绩 不 为 Null 的 记录 即 可 理解 
为 有 成 绩 记 录 ， 可 以 找 出 这 部 分 有 成 绩 记 录 的 学 生 编号 ; 如 果 学 生 的 成 绩 编号 不 在 这 
部 分 编号 里 ， 我 们 可 以 理解 为 该 学 生 是 没有 成 绩 记 录 信 息 的 。 


【问题 分 析 】 

本 练习 主要 是 学 习 如 何 使 用 子 查询 。 

【参考 步骤 】 

(1) 查询 所 有 有 成 绩 记录 的 学 生 编号 ，SQL 语句 如 下 。 
select SID from GradeInfo where Grade is not NULL:; 

(2) 查询 不 在 上 述 学 生 编号 中 的 学 生 信息 。 


mysql> select * from StuInfo where SID not in(select SID from GradeInfo where Grade is not 
NULD); 


查询 结果 如 图 3-26 所 示 。 


单元 三 “使 用 SQL 语句 


Inysql> select # from Stulnfo where SID not inCselect SID from GradeInfo where Gr| 
lade is not NULL 


wumao 
maomao 


其 实 这 个 查询 也 可 以 使 用 多 表 联 合 查询 或 其 他 的 方法 实现 ， 请 自行 尝试 。 


练习 3: 查 询 所 有 成 绩 在 85 分 以 上 的 成 绩 信息 及 对 应 的 学 生 姓 名 。 


【问题 描述 】 


成 绩 记 录 表 中 的 Grade 字段 记录 了 学 生成 绩 ， 可 以 在 其 中 找到 成 绩 在 85 分 以 
上 的 成 绩 信息 。 但 是 该 表 中 没有 学 生 姓 名 的 字段 记录 ， 只 能 根据 学 生 编号 找到 学 生 
的 姓名 ， 这 需要 使 用 到 左 连接 


【问题 分 析 】 
本 练习 主要 是 学 习 如 何 使 用 左 外 连接 
【参考 步骤 】 


(1) 实现 该 查询 的 SQL 语句 如 下 。 


Imysql> select GID,GradeInfo.SID,SubID,Grade,Remark,StuInfo.Name 
-> from Gradeinfo inner join StuInfo 
-> on Gradeinfo.SID=StuImfo.SID 


-> Where Gradeinfo.Grade>=85; 


(2) 以 上 查询 运行 的 效果 如 图 3-27 所 示 


1> select GID.GradeInfo-SID.SubID.Gradqe-.Remark-.StuInfo.Name 
> from Gradeinfo inner join Stulnfo 
on Gradeinfo-SID=StuInfo.SID 


where Gradeinfo -Gradey> 


图 3-27 
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令 第 二 阶段 多 


练习 1: 查询 所 有 成 绩 在 70 分 到 90 分 之 间 的 成 绩 信 息 。 


可 以 参考 的 SQL 语句 如 下 。 
mysql> select * 位 om gradeinfo where grade between 70 and 90; 
请 思考 如 何 用 其 他 的 方法 实现 。 
练习 2: 计算 每 个 学 生 的 考试 科目 的 平均 成 绩 。 
可 以 参考 的 SQL 语句 如 下 。 
select SID,avg(Grade) as 平均 成 绩 ' ffom GradeInfo group by SID; 
【拓展 作业 】 
1. 查询 所 有 学 生 信息 及 对 应 的 班级 名 称 。 
2. 查询 姓名 中 包含 有 “mao” 的 学 生 信息 及 对 应 的 成 绩 信 息 。 


3. 计算 每 个 学 生 的 考试 科目 的 平均 成 绩 ， 并 将 平均 成 绩 前 两 名 的 学 生 信息 显示 
出 来 。 


时 课程 目标 


BP 理解 视图 
BP 创建、 编辑、 删除 视图 
BP 使 用 视图 
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计算 机 数据 库 中 的 视图 是 一 个 虚拟 表 ， 其 内 容 由 查询 定义 。 同 真实 的 表 一 样 ， 视 
图 包含 一 系列 带 有 名 称 的 列 和 行 数 据 。 但是， 视图 并 不 在 数据 库 中 以 存储 的 数据 值 集 
形式 存在 。 行 和 列 数据 来 自由 定义 视图 的 查询 所 引用 的 表 ， 并 且 在 引用 视图 时 动态 生 
成 。 视 图 是 一 个 虚拟 表 ， 其 内 容 由 查询 定义 。 同 真实 的 表 一 样 ， 视 图 的 作用 类 似 于 入 
选 。 定 义 视图 的 筛选 可 以 来 自 当 前 或 其 他 数据 库 的 一 个 或 多 个 表 ， 或 者 其 他 视图 。 视 
图 是 存储 在 数据 库 中 的 查询 的 SQL 语句 ， 它 主要 出 于 两 种 原因 : 一 是 安全 原因 ， 视 图 
可 以 隐藏 一 些 数据 ， 如 社会 保险 基金 表 ， 可 以 用 视图 只 显示 姓名 、 地 址 ， 而 不 显示 社 
会 保险 号 和 工资 数 等 ; 另 一 原因 是 可 使 复杂 的 查询 易于 理解 和 使 用 。 本 单元 重点 介绍 
视图 的 使 用 。 


4.1 视图 


视图 是 一 个 虚 表 。 视 图 并 不 包含 或 存储 真实 存在 的 数据 ， 它 只 是 对 一 个 或 者 多 个 
真实 存在 的 表 数 据 的 查询 。 


4.1.1 视图 优势 


对 于 一 个 已 经 创建 好 的 视图 ， 可 以 像 查询 一 个 普通 的 数据 表 一 样 对 它 进行 查询 。 
使 用 视图 有 以 下 几 个 优势 。 
将 对 数据 库 的 查询 集中 到 特定 的 数据 集中 。 事 先 把 需要 查询 的 结果 集 定义 到 
视图 中 ， 在 需要 该 数据 的 时 候 直 接 从 视图 中 查询 。 
简化 查询 操作 。 视 图 通常 是 基于 一 个 非常 复杂 的 查询 而 建立 ， 将 这 些 复 杂 的 
查询 定义 到 视图 中 ， 避 免 了 在 程序 中 创建 和 执行 这 些 复杂 的 查询 语句 。 
使 用 不 同 的 用 户 根据 自己 的 需要 对 相同 的 数据 进行 不 同 的 组 织 。 


4.1.2 视图 的 常见 应 用 


编写 视图 解决 查询 的 问题 不 是 必需 的 ， 但 是 可 以 为 查询 提供 极 大 的 方便 ， 下 面 列 
出 几 种 可 以 考虑 使 用 视图 的 情况 。 

e 当 某 条 SQL 语句 已 经 被 重用 了 很 多 次 时 。 

。 当 某 个 查询 非常 复杂 ， 在 编写 后 不 必 再 考虑 它 的 实现 细节 时 。 

。 当 我 们 需要 使 用 表 的 部 分 字段 而 不 是 全 部 时 。 

。 当 我 们 打算 展现 给 用 户 的 只 是 表 的 特定 部 分 而 不 是 全 部 时 。 

。 当 我 们 打算 修改 数据 显示 时 的 格式 及 表示 方式 时 。 
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在 视图 创建 以 后 ， 我 们 可 以 像 使 用 普通 表 那 样 使 用 它 。 可 以 对 其 进行 查询 、 过 滤 
和 排序 ， 或 者 将 视图 连接 到 其 他 的 表 或 视图 。 重 要 的 是 要 知道 视图 仅仅 是 用 来 查看 存 
储 在 别处 的 数据 表 的 一 种 设施 ， 它 本 身 并 不 包含 数据 ， 因 此 它们 返回 的 数据 是 从 其 他 
表 中 检索 出 来 的 。 在 这 些 真实 表 中 的 数据 发 生 改 变 时 ， 由 视图 查询 得 到 的 数据 也 将 发 
生 改 变 。 


4.1.3 ”视图 遵循 的 规则 


下 面 介绍 视图 在 创建 和 使 用 时 一 些 常见 的 应 该 遵循 的 规则 。 
视图 名 必须 唯一 (不 能 与 现 有 数据 表 或 现 有 的 数据 视图 同名 )。 
对 于 可 以 创建 的 视图 数目 没有 限制 。 
创建 视图 的 用 户 必 须 有 足够 的 权限 。 
视图 可 以 棋 套 ， 也 就 是 可 以 从 现 有 的 视图 创建 视图 。 
创建 视图 的 FROM 子 句 不 能 包含 子 查询 。 
视图 不 能 索引 ， 也 不 能 与 触发 器 关联 。 


4.1.4 创建 视图 


在 理解 了 什么 是 视图 以 及 视图 可 以 使 用 的 情况 后 ， 我 们 来 看 一 看 如 何在 MySQL 中 
创建 和 使 用 视图 。 
视图 的 创建 使 用 CREATE VIEW 语句 。 基 本 语法 如 下 。 


CREATE VIEW view_name 


[column1,…,columnN] 
AS 
select_statement 


其 中 , CREATE VIEW 是 关键 字 ; view_name 就 是 要 创建 的 视图 名 ;columnl 到 columnN 
是 为 视图 中 的 Select 语句 所 选择 的 列 对 应 的 别名 ， 也 可 以 在 Select 语句 中 指定 别名 ; 
select_statement 用 来 定义 视图 的 查询 语句 。 

示例 :建立 一 个 视图 ， 用 于 查询 借阅 记录 中 借阅 人 的 姓名 、 所 借 书籍 名 称 、 借 阅 
时 间 。 其 SQL 语句 如 下 。 


mysql> create view view_borrow 
->as 
-> select card.Name,books.BName,RDATE 
-> from borrow left join card on borrow.CNO=card.CNO 
-> left join books on borrow.BNO=books.BNO: 


视图 创建 成 功 以 后 ， 可 以 像 普 通 数据 表 一 样 对 其 进行 查询 ， 如 图 4-1 所 示 。 
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图 4-1 


4.1.5 修改 视图 


修改 视图 使 用 ALTER VIEW 语句 ， 语 法 如 下 。 


ALTER VIEW view_name 
[column1,……:columnN] 
AS 


select_statement 


其 中 ， 各 项 参数 的 含义 与 创建 视图 语法 的 各 项 参数 完全 相同 。 
4.1.6 删除 视图 


删除 视图 使 用 DROP VIEW 语句 ， 语 法 如 下 


DROP VIEW [IF EXISTS ] 


View_name [,view_name] ... 


该 语句 可 以 一 次 删除 一 个 或 者 多 个 视图 ， 可 选项 IF EXISTS 表示 当 存 在 时 进行 删 
除 ， 避 免 因 为 视图 不 存在 而 报错 。 


4.2 存储 过程 


迄今 为 止 ， 本 书 所 使 用 的 大 多 数 SQL 语句 都 是 针对 一 个 或 多 个 表 的 单条 语句 。 实 
际 中 ， 并 非 所 有 的 应 用 都 这 么 简单 ， 例 如 : 当 阅 览 室 中 的 某 一 本 书 被 借 出 时 ， 我 们 希 
望 除了 在 借阅 表 中 存储 相同 的 记录 的 同时 ， 能 把 书籍 信息 表 中 的 该 书籍 数量 进行 减 1 
处 理 。 这 显然 不 是 一 条 语句 可 以 执行 完成 的 。 

存储 过 程 是 使 用 SQL 语句 和 过 程控 制 语 句 编写 的 程序 ， 存 储 在 数据 库 服务 器 上 ， 
可 以 由 用 户 直 接 或 者 间接 地 进行 调用 。 其 中 , 过 程控 制 语句 指 变量 声明 、 赋 值 、 分 支 、 
循环 等 结构 控制 语句 。 
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4.2.1 创建 存储 过 程 


存储 过 程 可 以 非常 简单 ， 也 可 以 非常 复杂 ， 例 如 ， 按 照 某 一 个 复杂 的 业务 需求 
完成 一 系列 复杂 的 SQL 语句 。 创 建 存 储 过 程 通过 CREATE PROCEDURE 语句 实现 ， 
其 语法 如 下 。 


CREATE PROCEDURE proc_name 

([Ppro_parameter[,...] ] ) 

BEGIN 

routine_body 

END 
其 中 ，proc_name 为 要 创建 的 存储 过 程 名 ; ( [ pro_parameter[,…] ] ) 为 存储 过 程 的 参数 
列表 ， 多 个 参数 之 间 以 逗号 隔 开 ， 即 使 没有 参数 也 应 该 写 “0”; BEGIN、END 分 别 为 开 
始 和 结束 标志 ; routine_ body 为 定义 存储 过 程 操 作 的 语句 块 ， 语 句 块 由 SQL 语句 和 其 
他 过 程控 制 语句 组 成 。 

创建 存储 过 程 的 语句 中 ， 每 一 个 参数 pro_parameter 由 如 下 的 格式 定义 。 


parameter_ name [ INIOUTIINOUT ] datatype 


其 中 ，parameter name 是 参数 名 ; IN 指 该 参数 为 输入 参数 ， 用 于 向 存储 过 程 输入 值 ， 
如 果 定 义 ， 在 调用 存储 过 程 的 时 候 要 给 该 参数 传 参 OUT 表示 输出 参数 ， 调 用 存储 过 
程 后 可 以 通过 该 参数 得 到 值 ;, INOUT 指 输入 输出 型 参数 ,在 调用 存储 过 程 时 可 以 传 值 
给 存储 过 程 ， 在 存储 过 程 执行 过 程 中 ， 可 以 改变 其 值 ， 并 在 最 终 执 行 完 毕 后 可 以 返回 
值 给 调用 者 ，datatype 是 指 参数 的 数据 类 型 。 

CREATE PROCEDURE 语句 块 中 的 每 一 条 SQL 语句 都 是 以 “; ”分 号 作为 分 隔 符 
的 ， 而 MySQL 命令 行程 序 也 是 使 用 “; ”分 号 作为 分 隔 符 的 。 这 样 如 果 存 储 过 程 的 语 
句 块 中 再 出 现 以 “; ”分 号 结束 的 分 隔 符 时 ， 必然 会 使 用 存储 过 程 的 创建 而 出 现 句 法 错 
误 。 解 决 的 办 法 就 是 临时 更 改 命 令 行 程序 的 语句 分 隔 符 。 可 以 使 用 “DELIMITER //” 
将 命令 行程 序 的 分 隔 符 从 “; ”分 号 改 为 “/” 两 斜 杠 。 

示例 : 下 面 创建 一 个 最 简单 的 无 参 无 返回 值 的 存储 过 程 ， 用 于 查询 所 有 的 借阅 记 
录 ， 其 SQL 语句 如 下 所 示 。 


mysql> delimiter // 
mysql> create procedure proc_borrow 
= 
-> begin 
-> select * 位 om borrow: 
-> end 
->// 
Query OK, 0 rows affected (0.00 sec) 
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mysql> delimiter ; 
此 时 ， 创 建 以 proc_borrow 命令 的 存储 过 程 完成 了 。 但 是 要 记得 在 创建 完成 后 把 
命令 行程 序 的 分 隔 符 由 “/” 重 新 改 回 为 “; ”分 号 。 
成 后 就 可 以 执行 该 存储 过 程 了 ， 调 用 存储 过 程 使 用 CALL 指令 
CALL 关键 字 后 面 紧 跟 存储 过 程 名 ;如 果 存 储 过 程 有 参数 的 要 求 ， 再 跟 上 人参 数值 或 者 
已 经 申明 的 变量 。 如 下 所 示 ， 执 行 存储 过 程 proc_borrow。 


mysql> call proc_borrow; 


执行 后 的 效果 如 图 4-2 所 示 。 


309 -89-19 98 
! 2809-89 


IQuery OK, @ rows affected (0.05 sec) 


Inysql> 


图 4-2 


此 存储 过 程 本 身 没 有 意义 ， 只 是 为 了 说 明 无 参 无 返回 值 的 存储 过 程 创建 的 方法 ， 
其 运行 效果 与 直接 执行 查询 运行 的 效果 完全 相同 


4.2.2 删除 存储 过 程 
存储 过 程 在 创建 之 后 ， 被 保存 到 服务 器 上 以 供 使 用 ， 直 至 被 删除 。 删 除 命令 用 于 
从 服务 器 中 删除 存储 过 程 。 语 法 如 下 : 
DROP PROCEDURE [IF EXISTS] sp_name 


其 中 ，sp_name 指 要 删除 的 存储 过 程 的 名 字 。 
4.2.3 ”有 参 无 返回 值 的 存储 过 程 


上 例 中 的 proc_borrow 只 是 一 个 简单 的 存储 过 程 , 其 意义 不 大 , 而 存储 过 程 一 旦 定义 ， 
其 语句 块 就 被 保存 到 了 服务 器 ， 不 易 在 使 用 时 修改 ， 因 此 ， 需 要 给 存储 过 程 一些 参 数值 ， 
以 便 扩展 。 下 面 给 出 一 种 有 参 无 返回 值 的 存储 过 程 以 供 读者 参考 。 

示例 : 创建 一 个 存储 过 程 , 用 于 查询 某 段 日 期 时 间 范 围 内 的 书籍 借阅 记录 。 其 SQL 
语句 如 下 。 
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mysql> delimiter // 
mysql> create procedure proc_borrow 
El 
-> in dl datetime, 
-> in d2 datetime 
->) 
-> begin 
-> select card.Name,books.BName,RDATE 
-> from borrow left join card on borrow.CNO=card.CNO 
-> left join books on borrow.BNO=books.BNO 
-> Where RDATE between dl and d2: 
-> end 
->// 


mysql> delimiter : 


假设 要 查询 2009 年 9 月 1 日 到 2009 年 9 月 19 日 之 间 的 借 


储 过 程 调用 语句 如 下 所 示 。 


mysql> call proc_borrow ("2009-9-1','2009-9-19"); 
运行 的 结果 如 图 4-3 所 示 


“22989-9-19 3 


99-99 699:99:98 ! 
9-89-19 969: 0 


9-99-19 8690:60:90 } 
3 rows in set (0.00 sec) 
Query Ok. @ rows affected (0.05 sec) 


Inysql> 


单元 四 ”高 级 对 象 


阅 记录 , 可 以 使 用 的 存 


图 4-3 


4.2.4 有 参 有 返回 值 的 存储 过 

如 果 希 望 存储 过 程 的 执行 能 有 
以 给 存储 过 程 加 一 个 返回 值 参数 。 
的 外 部 以 什么 值 赋 给 OUT 参数 ,在 存储 过 
i 新 赋值 。 换 句 话 讲 ， 给 OUT 关键 字 


-个 返回 值 ， 以 提 f 


ZA 


绍 已 


项 述 的 


以 用 于 在 存储 过 程 内 部 给 其 赋值 ， 
示例 : 


创建 一 个 存储 过 程 ， 用 于 接收 
见 的 SQL 语句 如 下 所 示 。 


-个 借阅 卡 卡号 ， 


mysql> delimiter // 


mysql> create procedure proc_borrow 
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共 给 将 来 的 程序 ; 
这 个 参数 以 OUT 关键 字 表 示 ， 
旦 的 内 部 使 用 都 是 NULL; 


使 用 ， 我 们 可 
并 且 无 论 存储 过 程 
除非 ， 存储 过 程 


数 传 值 是 没有 意义 的 。 但 是 该 参数 可 
并 且 这 个 值 是 可 以 带 出 存储 过 
返回 该 借阅 卡 共 借 书籍 多 


得 给 调用 端的 。 


使 用 PHP 开发 Web 应 用 程序 


-> (IN var_cno int,OUT var_count int) 
-> begin 
-> select count(1) into var_count from borrow where cno=var cno; 
-> end 
->// 
Query OK, 0 rows affected (0.00 sec) 


mysql> delimiter ; 
存储 过 程 创建 完成 后 ， 可 以 对 其 进行 调用 ， 调 用 的 语句 如 下 


mysql> set @id=1; 
Query OK, 0 rows affected (0.00 sec) 


mysql> set @count=0; 
Query OK, 0 rows affected (0.00 sec) 


mysql> call proc_borrow (@id,@count); 
Query OK, 0 rows affected (0.00 sec) 
其 中 ，set 语句 用 于 创建 一 个 用 户 变量 ; @id 是 用 户 变量 的 名 字 ,， 用 于 向 存储 过 程 传递 
-个 要 查询 的 借阅 卡 卡号 ; @count 用 于 存储 过 程 计 算 完成 后 传 回 一 个 值 , 起 到 接收 的 
作用 ， 所 以 其 初 值 为 0。 
存储 过 程 调用 完成 后 , 得 到 的 返回 值 被 变量 @count 接收 ,此 时 可 以 从 该 返回 值 中 
得 到 最 终 的 结果 ， 查 询 的 代码 如 下 


mysql> select @count as ' 该 用 户 借 阅 书籍 总 数 '; 


触发 器 是 一 种 触发 执行 的 程序 ， 它 是 定义 在 数据 表 中 ， 且 当 数 据 表 上 发 生 相 应 触 
发 事件 时 自动 执行 的 程序 。 它 有 两 个 显著 特点 : 一 是 它 是 一 个 事先 定义 好 了 的 程序 ， 
在 执行 之 前 需要 在 数据 表 中 定义 好 ; 二 是 爷 ey 与 其 他 的 程序 不 一 样 的 是 它 
能 在 某 个 事件 发 生 时 自动 执行 ， 而 其 他 程序 是 需要 调用 后 才能 执行 的 。 
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4.3.1 触发 事件 


MySQL 支持 如 下 3 种 类 型 的 触发 事件 。 
INSERT 事件 : 当 向 表 中 插入 记录 时 触发 执行 触发 程序 。 例 如 , 执行 “insert into 
tbl_name values(Vv1.v2,v3,…)” 的 INSERT 语句 时 。 
UPDATE 事件 ， 当 更 新 表 中 记录 时 触发 执行 触发 程序 。 例 如 ， 使 用 UPDATE 语 
句 更 新 记录 时 。 
DELETE 事件 : 当 删 除 表 中 记录 时 触发 执行 触发 程序 。 例 如 ， 使 用 DELETE 语 
句 删除 记录 时 。 

对 于 触发 程序 的 执行 时 间 ，MySQL 支持 以 下 两 种 类 型 。 

e BEFORE: 在 触发 该 触发 程序 的 语句 之 前 执行 。 例 如， 在 INSERT 语句 之 前 执 
行 INSERT 类 型 的 触发 程序 。 
AFTER: 在 触发 该 触发 程序 的 语句 之 后 执行 。 例 如 ， 在 DELETE 语句 之 后 执 
行 触发 程序 。 


4.3.2 创建 触发 器 


在 创建 触发 器 时 ， 需 要 给 出 如 下 4 条 信息 。 


e 唯一 的 触发 器 名 。 

e 触发 器 关联 的 表 。 

e 触发 器 在 什么 操作 下 响应 (应 该 在 DELETE、INSERT 或 UPDATE 中 的 什么 时 
候 执行 )。 

e 触发 器 在 什么 时 间 执 行 (该 操作 之 前 还 是 之 后 )。 

创建 触发 器 应 该 使 用 CREATE TRIGGER 语句 ， 其 语法 如 下 。 


CREATE TRIGGER trigger name {BEFORE | AFTER} {INSERT | UPDATE | DELETE} 
ON tbl name FOR EACH ROW trigger statement 


其 中 ，trigger_name 为 触发 器 名 称 ，tbl_name 为 触发 器 关联 的 表 的 名 称 ，trigger_statement 
为 触发 器 执行 的 语句 或 者 语句 块 , 如 果 为 多 条 语句 组 成 的 语句 块 , 可 以 使 用 BEGIN...END 
进行 定义 。 组 成 trigger_statement 的 语句 可 以 是 存储 过 程 中 允许 的 任何 语句 。 

示例 : 当 向 db_test 数据 库 中 的 card 表 中 新 增 一 个 借阅 用 户 时 ， 显 示 “ 新 增 用 户 
成 功 ”。 实 现 其 需求 的 创建 触发 器 SQL 语句 如 下 。 


mysql> delimiter // 
mysql> create trigger trg_card after insert on card 
-> for each row 
> begin 
-> select ' 已 经 增加 数据 ' into @ee; 
->end 


使 用 PHP 开发 Web 应 用 程序 


->// 
Query OK., 0 rows affected (0.48 sec) 


mysql> delimiter ; 
当 触 发 器 创建 成 功 后 , 触发 器 中 所 定义 的 SQL 语句 并 不 会 马上 执行 , 而 是 会 等 到 
向 card 表 中 插入 数据 后 ， 这 段 SQL 语句 才 会 执行 。 运 行 以 下 的 INSERT 语句 向 表 中 
增加 数据 。 


mysql> insert into card(name,cname) values(' 张 三 '" 白 动 化 2 班 ); 
Query OK, 1 row affected (0.33 sec) 


对 表 中 的 数据 进行 查询 ， 结 果 如 图 4-5 所 示 。 


图 4-5 
成 功 增加 数据 后 ， 查 询 触 发 器 中 定义 的 “@ee” 变 量 ， 可 以 发 现 ，@ee 变量 的 值 已 经 
发 生 改变 ， 如 图 4-6 所 示 。 


图 4-6 


以 上 示例 充分 说 明了 触发 器 中 事先 定义 好 的 代码 只 有 在 触发 器 的 触发 条 件 满足 时 
才 会 执行 。 在 任何 其 他 时 候 都 不 会 执行 。 所 以 说 ， 触 发 器 是 一 种 触发 执行 程序 。 


4.3.3 删除 触发 器 


删除 触发 器 的 语法 如 下 。 
DROP TRIGGER trigger_name 


器 的 名 字 。 至 此 ， 我 们 已 经 


其 中 ，trigger_name 指 触 


语法 ， 在 此 对 于 删除 触发 器 就 不 再 示例 了 ， 请 读者 自行 实验 。 的 是 ， 触 发 器 
不 能 更 新 或 覆盖 。 为 了 修改 一 个 触发 器 ， 必 须 先 删除 它 ， 然 后 再 重新 创建 。 


4.3.4 _ INSERT 触发 器 


INSERT 触发 器 在 INSERT 语句 执行 之 前 或 之 后 执行 。 需 要 知道 以 下 几 点 。 
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e 在 INSERT 触 发 器 代码 内 ， 可 引用 一 个 名 为 NEW 的 虚拟 表 , 访问 被 插入 的 行 


数据 。 
e 在 BEFORE INSERT 触发 器 中 ，NEW 中 的 值 也 可 以 被 更 新 (允许 更 改 被 插入 
的 值 )。 


e 对 于 AUTO INCREMENT 列 ，NEW 在 INSERT 执行 之 前 包含 0， 在 INSERT 
执行 之 后 包含 新 的 自动 生成 值 。 
示例 : 当 阅 览 室 的 借阅 人 借阅 一 本 书 时 ,会 在 borrow 表 产生 一 条 借阅 记录 (INSERT 
操作 )， 此 时 被 借阅 的 书籍 的 总 数量 应 该 减 1。 利 用 触发 器 实现 当 borrow 表 的 insert 操 
作 发 生 时 ， 自 动 把 books 表 的 该 书籍 总 数 quantity 减 1。 利 用 MySQL 的 触发 器 实现 的 
SQL 语句 如 下 。 


mysql> delimiter // 
mysql> create trigger trg_borrow_insert 
-> after insert on borrow 
-> for each row 
-> begin 
-> select new.bno into @b; 
-> Update books set quantity = quantity - 1 where bno=(@b; 
-> end 
->// 
Query OK, 0 rows affected (0.12 sec) 
mysql> delimiter ; 


从 上 面 代 码 中 看 到 ，new.bno 指向 borrow 数据 表 中 新 插入 的 值 的 bno 列 的 值 。 这 
就 是 利用 了 NEW 关键 字 的 NEW.column_name 形式 引用 新 数据 记录 中 某 列 的 值 。 

使 用 NEW 时 ， 如 果 是 定义 BEFORE 触发 程序 ， 并 且 有 UPDATE 权限 ， 可 以 使 用 
“SET NEW.col name=value” 的 方式 修改 它 的 值 。 也 就 是 说 ， 可 以 使 用 触发 程序 来 修 
改 将 要 插入 到 新 行 中 的 值 ， 或 用 于 更 新 新 行 的 值 。 

当 以 上 的 触发 器 定义 好 后 ， 尝 试 添加 一 条 借阅 记录 : 


mysql> insert into borrow(cno,bno,rdate) values(3,1,now()); 
Query OK, 1 row affected (0.07 sec) 


通过 查询 借阅 记录 表 borrow 可 以 发 现 该 借阅 记录 已 经 成 功 添加 ， 如 图 4-7 所 示 。 
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再 查询 书籍 记录 表 books 可 以 发 现 书号 为 1 的 书籍 总 数 减 1 了， 如 图 4-8 所 示 。 


AUTHOR 


4.3.5 _ DELETE 触发 器 


DELETE 触发 器 在 DELETE 语句 执行 之 前 或 之 后 执行 。 使 用 DELETE 触发 器 , 需 
要 知道 以 下 两 点 。 
e 在 DELETE 触发 器 代码 内 ， 可 以 引用 一 个 名 为 OLD 的 虚拟 表 ， 从 OLD 表 可 
以 访问 到 被 删除 的 行 的 数据 
e OLD 的 值 全 都 是 只 读 的 ， 不 能 更 新 
示例 : 对 db_test 数据库 中 的 borrow 数据 表 建 立 DELETE 触发 器 ， 当 删除 一 条 借 
阅 记 录 时 , 该 借阅 记录 的 书籍 号 所 对 应 的 书籍 数量 加 1。 实 现 该 需求 的 SQL 代码 如 下 。 


mysql> delimiter // 
mysql> create trigger trg_borrow_delete 
-> after delete on borrow 
-> for each row 
-> begin 
-> select OLD.bno into @b; 
-> Update books set quantity=quantity+1 where bno=(@b; 
-> end 
->// 
Query OK, 0 rows affected (0.10 sec) 


mysql> delimiter ; 


以 上 代码 中 定义 了 如 果 要 对 borrow 数据 表 进 行 删除 操作 ， 由 OLD 读 出 要 删除 的 
这 条 借阅 记录 是 借 了 哪 一 本 书 (书籍 编号 是 多 少 )， 再 根据 书籍 编号 将 对 应 的 书籍 数量 
做 加 1 处 理 。 

创建 好 触发 器 后 ， 执 行 删除 语句 如 下 。 
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mysql> delete from borrow where ID=7; 
Query OK, 1 row affected (0.08 sec) 


在 完成 删除 操作 后 ， 重 新 查询 borrow 数据 表 ， 该 借阅 记录 已 经 删除 ， 如 图 4-9 所 示 。 


图 4-9 


对 书籍 信息 进行 查询 ， 效 果 如 图 4-10 所 示 。 


图 4-10 
由 图 可 知 ， 删 除 的 借阅 记录 所 对 应 的 书籍 数量 已 经 增加 了 。 至 此 ， 完 成 了 db test 数 
据 库 中 borrow 数据 表 DELETE 触发 器 的 创建 。 


4.3.6 UPDATE 触发 器 


UPDATE 触发 器 在 UPDATE 语句 执行 之 前 或 之 后 执行 。 需 要 下 几 点 。 
e 在 UPDATE 和 触发 器 代码 中 ， 可 以 引用 一 个 名 为 OLD 的 虚拟 表 访 问 以 前 
(UPDATE 语句 前 ) 的 值 ， 引 用 一 个 名 为 NEW 的 虚拟 表 访 问 ei 
e 在 BEFORE UPDATE 触发 器 中 ，NEW 中 的 值 也 可 以 被 更 新 (允许 更 改 将 要 用 
于 UPDATE 语句 中 的 值 )。 
e OLD 中 的 值 全 都 是 只 读 的 ， 不 能 更 新 
实际 上 ，UPDATE 操作 可 以 理解 为 “删除 + 新 增 ” 并 存 的 一 种 操作 ,所 以 UPDATE 
触发 器 也 可 以 理解 为 同时 具有 NEW 和 OLD 两 个 虚拟 表 的 触发 器 。 以 前 两 节 示例 实验 
效果 为 基础 ， 请 读者 自行 实验 UPDATE 触发 器 
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4.3.7 ”关于 触发 器 


请 记 住 触发 器 以 下 几 个 重点 。 

e 与 其 他 DBMS 相 比 ，MySQL 中 支持 的 触发 器 相当 初级 。 例 如 ， 它 没有 类 似 于 
MS-SQL 的 列 级 触发 器 。 
创建 触发 器 需要 特殊 的 安全 访问 权限 但是， 触发 器 的 执行 是 自动 的 ， 与 权 
限 无 关 。 
应 该 用 触发 器 来 保证 数据 的 一 致 性 (大 小 写 、 格 式 等 )。 在 触发 器 中 执行 这 种 类 
型 的 应 用 的 优点 是 它 总 是 进行 这 种 处 理 ， 而 与 客户 机 应 用 无 关 。 
触发 器 的 一 种 非常 有 意义 的 使 用 是 创建 审计 跟踪 。 使 用 触发 器 ， 把 更 改 (甚至 
之 前 之 后 的 状态 ) 记 录 到 另 一 个 表 非 常 容易 。 
MySQL 触发 器 中 不 支持 CALL 语句 。 这 表示 不 能 从 触发 器 内 调用 存储 过 程 。 
所 需 的 存储 过 程 代码 需要 复制 到 触发 器 内 。 


【单元 小 结 】 


。 了 解 视图 的 作用 ， 掌 握 如 何 创建 、 删 除 、 修 改 视图 
。 掌握 创建 、 删 除 存储 过 程 及 参数 、 返 回 值 复杂 的 存储 过 程 
e 理解 触发 器 概念 ， 创 建 、 删 除 几 种 不 同 的 触发 器 


【单元 自 测 】 


1. 关于 视图 ，( ， ) 说 法 是 错误 的 。 
A. 使 用 视图 ， 可 以 简化 数据 的 使 用 
B. 使 用 视图 ， 可 以 保护 敏感 数据 
C. 视图 是 一 种 虚拟 表 ， 视 图 中 的 数据 只 能 来 源 于 物理 数据 表 ， 不 能 来 源 于 其 他 视图 
D. 视图 中 只 存储 了 查询 语句 ， 并 不 包含 任何 数据 
2. 银行 系统 中 有 账户 表 和 交易 表 ， 账 户 表 中 存储 了 各 存款 人 的 账户 余额 ,交易 表 
中 存储 了 各 存款 人 每 次 的 存 取款 金额 。 为 保证 存款 人 每 进行 一 次 存 、 取 款 交 易 ， 都 正 
确 地 更 新 了 该 存款 人 的 账户 余额 ， 以 下 选项 中 正确 的 做 法 是 (  )。 
A. 在 账户 表 上 创建 msert 触发 器 B. 在 交易 表 上 创建 insert 触发 器 
C. 在 账户 表 上 创建 检查 约束 D. 在 交易 表 上 创建 检查 约束 
. 下 面 关 于 存储 过 程 的 描述 不 正确 的 是 (。” ”)。 
A. 存储 过 程 实际 上 是 一 组 TSQL 语句 
B. 存储 过 程 预 先 被 编译 存放 在 服务 器 的 系统 中 
C. 存储 过 程 独 立 于 数据 库 而 存在 
D. 存储 过 程 可 以 完成 某 一 特定 的 业务 逻辑 


元 四 高 级 对 象 


4. 可 以 创建 视图 的 语法 是 ( 。 )。 
A. CREATE TRIGGER B. CREATE DATABASES 
C. CREATE VIEW D. CREATE PROCEDURE 
- 可 以 创建 触发 器 的 语法 是 (。”)。 
A. CREATE TRIGGER B. CREATE DATABASES 
C. CREATE VIEW D. CREATE PROCEDURE 


【上 机 实战 】 


上 机 目标 
。 熟练 掌握 视图 的 使 用 
。 熟练 掌握 触发 器 的 使 用 
。 熟练 掌握 存储 过 程 的 使 用 


上 机 练习 


令 第 一 阶段 多 


练习 1: 创建 一 个 视图 ， 用 于 查询 学 生 的 基本 信息 ， 要 求 查询 出 
来 的 信息 中 包括 班级 名 称 字段 。 


【问题 描述 】 


创建 视图 view_stuinfo 查询 所 有 学 生 的 基本 信息 , 视图 包括 的 字段 有 SID、Name、 
Age、Sex、CName。 


【问题 分 析 】 


本 练习 主要 学 习 如 何 创 建 一 个 视图 。 
学 生 信息 表 中 并 不 包括 班级 名 称 字 段 ， 只 有 在 班级 信息 表 中 才 有 该 字段 。 因 此 ， 
本 视图 在 创建 过 程 中 需要 对 学 生 信 息 表 和 班级 信息 表 进 行 连 接 查询 。 


【参考 步骤 】 
(1) 编写 如 下 的 SQL 语句 ， 用 以 创建 视图 。 
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mysql> create view view_stuinfo as 
-> select SID,Name,Age,Sex,CName 
-> from stuinfo left join clsinfo 
-> on stuinfo.CID=clsinfo.CID: 
Query OK, 0 rows affected (0.10 sec) 


(2) 对 该 创建 好 的 视图 做 查询 ， 如 图 4-11 所 示 。 


19 ! 


《B.B6 sec) 


图 4-11 


练习 2: 创建 一 个 存储 过 程 ， 用 于 向 stuinfo 表 插 入 学 生 信息 。 


【问题 描述 】 


向 stuinfo 表 增 加 学 生 信息 ， 存 储 过 程 的 参数 应 该 有 : 学 生 姓 名 、 年 龄 、 性 别 、 月 


在 班级 编号 。 因 为 增加 完成 后 无 须 返 回 ， 所 以 存储 过 程 可 以 没有 返回 值 。 
【问题 分 析 】 
本 练习 主要 学 习 如 何 创建 一 个 有 参 无 返回 值 的 存储 过 程 
【参考 步骤 】 
(1) 编写 如 下 的 SQL 语句 ， 用 以 创建 存储 过 程 


mysql> delimiter // 
mysql> create procedure proc_stuinfo 
-> ( 
-> inVv_Name varchar(20), 
->inVv_Age int(4), 
->inV_Sex varchar(20), 
->inv_ CID int(4) 
->) 
-> begin 
->insert into stuinfo(Name,Age,Sex.CID) 
values(v_Name,v Age,vy_Sex,v_CID); 
-> end 
->// 
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Query OK, 0 rows affected (0.40 sec) 


mysql> delimiter ; 


(2) 在 存储 过 程 创建 完成 后 , 可 以 对 该 存储 过 程 使 用 call 指令 进行 调用 , 如 图 4-12 
所 示 。 


图 4-12 


(3) 在 执行 完 存 储 过 程 后 ， 对 表 进 行 查询 ， 可 以 发 现 表 中 已 经 添加 了 刚才 使 用 存 
储 过 程 添加 进来 的 数据 。 


练习 3: 创建 一 个 触发 器 , 用 于 对 gradeinfo 中 的 数据 进行 修改 时 ， 
对 数据 所 做 的 修改 信息 都 保留 到 Remark 字段 中 。 


【问题 描述 】 
此 触发 器 的 创建 要 在 修改 数据 之 前 进行 ， 在 修改 数据 之 前 把 NEW 数据 表 中 的 
Remark 字段 的 内 容 改 为 修改 数据 的 时 间 。 从 而 将 修改 信息 记录 到 新 的 数据 表 中 。 


【问题 分 析 】 

本 练习 主要 学 习 如 何 创建 一 个 触发 器 。 

【参考 步骤 】 

(1) 编写 如 下 的 SQL 语句 ， 用 以 创建 该 表 上 的 存储 过 程 


mysql> delimiter // 
mysql> create trigger trg_ gradeinfo _ update 
-> before update on gradeinfo 
-> for each row 
-> begin 
-> set new.remark=concat(date_format(now(),%Y-%m-%d %k:%i:%s"),' 修 改 了 数据 "); 
-> end 
->// 
Query OK, 0 rows affected (0.10 sec) 
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(2) 在 触发 器 创建 完成 后 ， 我 们 可 以 看 到 ， 查 询 出 来 的 数据 如 图 4-13 所 示 。 


Lb 时 lect * from gradeinfoj 


(3) 此 时 ， 对 GID=8 的 同学 的 数据 进行 更 新 ， 如 图 4-14 所 示 。 


Inysql> update gradeinfo set grade=8B where gid=8; 
Query OK, 1 row affected ‘@ 
Rows matched: 1 Changed: 1 Warni 


图 4-14 
(4) 更 新 完成 后 再 对 表 进 行 一 次 查询 ， 结 果 如 图 4-15 所 示 。 


图 4-15 


(5) 从 图 中 可 以 看 到 ， 触 发 器 可 以 自动 执行 ， 因 此 任何 时 候 修改 成 绩 数据 都 是 有 
记录 的 。 


【拓展 作业 】 


1. 新 建 一 个 查询 成 绩 信 息 的 视图 ， 要 求 显 示 的 字段 如 下 。 
成 绩 编号 、 姓 名 、 科目 名 你 、 成 绩 

新 建 一 个 用 于 新 增 成 绩 信 息 的 存储 过 程 ， 要 求 
输入 参数 : 学 生 编 号 、 科 目 编号 、 成 绩 
输出 参数 : 无 

新 建 一 < 个 下 有 -个 人 的 平均 成 绩 的 存储 过 程 ， 要 求 的 参数 如 下 。 


[2 


的 参数 如 下 。 
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4. 新 建 一 个 用 于 统计 班级 平均 成 绩 的 存储 过 程 ， 要 求 的 参数 如 下 。 
输入 参数 : 班级 编号 


输出 参数 : 该 班级 的 平均 成 绩 


5. 创建 一 个 删除 班级 信息 的 触发 器 ， 当 删除 班级 信息 时 ， 先 自动 将 该 班级 所 有 学 
生 和 学 生 的 成 绩 信息 删除 。 
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外 和 中 


近年 来 ， 随 着 网 络 技 术 的 发 展 ,动态 网 站 技术 呈现 出 百家争鸣 的 景象 。 微软 公 
司 推出 的 Active Server Pages(ASP) 和 基于 .Net 平台 的 ASPNET 是 基于 Windows 平 
台 的 动态 网 站 开发 技术 ， 以 及 是 由 Sun Microsystems 公司 倡导 , 许多 公司 参与 一 起 
建立 的 一 种 动态 网 页 技术 标准 JSP(Java Server Pages)。 由 Zend 公司 所 推出 的 PHP 
也 成 为 另 一 门 跨 平 台 的 动态 网 站 开发 语言 。 

PHP， 是 英文 超级 文本 预 处 理 语言 Hypertext Preprocessor 的 缩写 。PHP 是 一 种 
HTML 内 谈 式 的 语言 ， 是 一 种 在 服务 器 端 执行 的 谈 入 HIML 文档 的 脚本 语言 ,语言 的 
风格 类 似 于 C 语言 ， 主 要 用 于 处 理 动态 网 页 。 它 是 当今 Internet 上 最 为 火热 的 脚本 语 
言 。 它 也 包含 命令 行 执 行 接口 (Command Line Interface) 或 用 于 创建 图 形 用 户 界面 (GUI) 
的 程序 。 


5.1 PHP 开发 环境 和 配置 


学 习 PHP 需要 一 个 开发 环境 。 由 于 PHP 是 一 门 服务 器 端 脚本 语言 ， 因 此 在 学 习 
时 需要 一 个 可 发 布 PHP 脚本 的 Web 服务 器 。Web 服务 器 有 很 多 ， 如 基于 Windows 平 
台 的 IIS、 基于 UNIX/Linux 平台 的 Apache 服务 器 等 。 当然, Apache 也 有 基于 Windows 
平台 的 版 本 。 

安装 PHP 环境 , 首先 需要 获取 Web 服务 器 和 PHP 的 安装 包 。 本 书 以 Apache 服务 
器 为 例 ， 讲 解 PHP 开发 环境 的 安装 。Apache 服务 器 和 PHP 都 是 免费 的 ， 并 且 也 是 开 
放 源 码 的 ， 因 此 可 以 从 其 官方 网 站 免费 下 载 。 


5.1.1 安装 Apache 服务 器 


Apache 服务 器 是 最 为 流行 的 Web 服务 器 之 一 ， 可 以 运行 在 几乎 所 有 的 计算 机 平 
台 上 。 它 是 由 NCSA 服务 器 发 展 而 来 的 。Apache 最 初 是 UNIX 系统 上 的 一 个 服务 器 ， 
其 设计 目标 是 建立 一 个 全 功能 、 高 效率 的 Web 服务 器 。 由 于 其 免费 、 开 放 源 码 性 ， 
Apache 服务 器 的 发 展 相当 迅速 。 

大 家 可 以 在 其 官方 网 站 http://httpd.apache.org/download.cgi 下 载 最 新 版 本 。 下载 的 
Apache 服务 器 分 为 两 种 : 一 种 是 源码 包 ， 一 种 是 安装 包 。 如 果 是 在 UNIX/Linux 系统 
上 安装 ， 可 以 选择 下 载 源码 包 ， 将 下 载 后 的 源码 包 在 UNIX/Linux 平台 上 进行 编译 即 
可 。 若 是 在 Windows 系统 上 进行 安装 ， 可 下 载 其 已 编译 好 的 安装 包 。 

通常 下 载 的 Apache 安装 包 文件 名 类 似 为 httpd-2.2.21-win32-x86-no_ssl.msi。 其中， 
2.2.21 为 Apache 服务 器 的 版 本 号 ; win32-x86 表示 该 安装 包 只 能 在 Windows 平台 下 安 
装 ; no_ssl 表示 该 安装 包 不 包含 安全 套 接 层 (SSL) 协 议 。 目 前 最 稳定 的 新 的 可 用 版 本 为 
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Apache2.2.21， 本 书 也 将 采用 该 版 本 。 
下 载 完成 后 ， 安 装 步骤 如 下 。 
第 一 步 : 双击 下 载 的 Apache 安装 包 ， 启 动 Apache 安装 向 导 ， 如 图 5-1 所 示 。 
Asche MTTP Server 2.21 = 


Welcome to the Installation Wizard 
Apache HTTP Server 2.2.21 


The Instalation Wizard wil install Apache HTTP Server 2.2.21 
on your computer, To continue, didk Next 


WARNING: This program is protected by copyright law and 
Nternational treaties. 


第 二 步 : 在 “Installation Wizard” 对 话 框 中 ， 单 击 “Next” 按 钮 ， 弹 出 “License 
Agreement” 界 面 ， 如 图 5-2 所 示 。 


License Agreement 
Please read the folowing license agreement carefuly. 


Apache License 
Version 2.0, January 2004 
http:/www.apache.org/licenses/ 
RMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION 


1. Definitions. 


“License” shall mean the terms and conditions for use, reproduction, and 
distribution as defined by Sections 1 through 9 of this document 


图 5-2 

第 三 步 : 继续 单 击 “Next” 按 钮 ， 直 到 出 现 “Server Information ”界面 ， 效 果 如 
图 5-3 所 示 ， 大 家 安装 时 直接 按 图 所 示 填 写 就 行 了 。 这 里 “Network Domain” 表 示 域 
名 ,“Server Name” 表 示 服 务 器 名 称 ,“Administrators Email Address ”表示 管理 员 的 
电子 邮件 。 第 一 个 单 选 按钮 要 求 选择 所 安装 的 Apache 服务 器 是 否 让 本 计算 机 的 所 有 
用 户 使 用 ， 并 将 Apache 服务 器 安装 为 Windows 服务 ,这 是 Apache 推荐 的 方式 。 若 不 
想 让 当前 计算 机 的 所 有 用 户 使 用 ， 可 点 选 第 二 个 单 选 按钮 ， 则 只 用 于 当前 用 户 ， 并 且 
需 手 动 启动 所 安装 的 Apache 服务 器 ， 其 端口 是 8080。 
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Please enter your server's nformation. 


Network Domain (e.g. somenet.com) 


Server Name (e.g. www.somenet.com): 


ocahost | 


Administrator's Emal Address (e.g. webmaster @somenet.com): 
estesnaom | 
Instal Apache HTTP Server 2.2 programs and shortcuts for: 


(for Al Users, on Port 80, as a Service —Recommended. 
only for the Current User on Port 8080, when started Manually. 


ce Fe em | 
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Installshield 


第 四 步 : 继续 单 击 “Next” 按 钮 ， 完 成 后 可 在 桌面 右 下 角 系 统 状态 栏 显示 Apache 
运行 图 标 同 |， 至 此 ，Apache 服务 器 安装 完毕 。 
如 果 遇 到 Apache 无 法 启动 的 情况 ， 则 通常 都 是 因为 其 他 应 用 程序 占用 了 ! 
80 端口 , 所 以 需要 检查 是 否 有 正在 运行 占用 80 端口 的 软件 , 如 迅雷 、PPLIVE、; 


快车 等 程序 。 尝 试 关闭 这 些 程序 重新 启动 Apache 服务 。 


om 一 


新 安装 的 Apache， 其 网 站 根 目录 为 Apache 安装 目录 下 的 名 为 htdocs 的 目录 ,将 
需要 运行 的 网 页 文件 放 入 该 目录 。 


5.1.2 安装 PHP 


PHP 包 提 供 了 对 PHP 脚本 文件 进行 编译 、 解 释 等 功能 ， 可 以 通过 其 官方 网 站 
http://www.php.net 免费 获得 。PHP 官方 网 站 提供 了 3 种 PHP 包 的 下 载 : 第 一 种 是 源码 
包 ; 第 二 种 是 WindowsZIP 包 ; 第 三 种 是 Windows 安装 包 。 

若是 在 Windows 平台 安装 PHP， 可 选择 下 载 ZIP 压缩 包 或 是 Windows 安装 包 进 
行 安装 。 若 是 在 UNIX/Linux 系统 上 安装 PHP， 可 下 载 源 码 包 直接 进行 编译 。 

安装 完成 后 修改 Apache 配置 文件 ， 打 开 Apache 安装 目录 下 conf 目录 中 的 
httpd.conf 文件 。 打 开 httpd.conf 文件 ， 找 到 如 下 代码 。 


#L.oadModule ssl module modules/mod ssl.so 
在 该 行 下 添加 如 下 内 容 。 
LoadModule php5_ module c:/php/php5apache2.dll 


AddType application/x-httpd-php.php 
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PHPIniDir "ce:\php" 


重新 启动 Apache 服务 器 。 至 此 ，Apache 服务 器 和 PHP 安装 完成 。 


5.1.3 ”安装 整合 套件 WAMP 


前 面 讲 到 的 PHP 开发 环境 的 安装 ， 需 要 先 安装 Apache 服务 器 ， 然 后 再 安装 PHP。 
实际 上 在 Windows 平台 上 有 一 种 较为 简单 的 方式 ， 就 是 直接 使 用 已 经 包含 Apache 服务 
器 和 PHP 的 整合 套件 。 

WAMP 是 一 个 只 能 用 于 Windows 平台 的 整合 套件 ， 可 通过 其 官方 网 站 
“http://www.wampserver.com/en/” 下 载 ， 下 载 安装 完成 后 在 屏幕 右 下 角 系 统 状 态 栏 中 将 
显示 图 标 国 |， 在 浏览 器 地 址 栏 输入 “http:/127.0.0.1”， 将 显示 类 似 如 图 5-4 所 示 的 


Version 1.7.4-version 
PHPS-Apehe2- MySQLS frangalse 


5.2 开发 工具 


使 用 记事 本 是 在 进行 PHP 开发 中 最 简单 的 一 种 开发 方式 , 但 是 选择 一 个 具有 相关 
功能 的 编译 器 或 合适 的 集成 开发 环境 (IDE)， 无 疑 会 使 你 有 事半功倍 的 效果 。 
目前 常用 的 开发 工具 有 以 下 几 种 。 


使 用 PHP 开发 Web 应 用 程序 


EditPlus 

NotePad++ 
Eclipse+PDT 
ZendStudio for Eclipse 
Dreamweaver 


5.3 第 一 个 PHP 程序 ; HELLO,WORLD! 


创建 一 个 文本 文件 ， 再 将 文件 名 改 为 hellophp， 在 该 文件 中 写 入 以 下 代码 : 


<?php 
echo "HELLO,WORLD!": 
es 


其 中 第 一 行 和 第 三 行为 PHP 脚本 标记 , 其 含义 是 通知 服务 器 标记 内 的 脚本 需要 以 
PHP 引擎 解释 该 脚本 。 中 间 为 PHP 脚本 ，echo 表示 将 输出 其 后 的 内 容 。 

输入 完成 后 ， 保 存 该 文件 ， 将 该 文件 保存 到 Apache 安装 目录 下 的 htdocs 目录 中 ， 
然后 在 浏览 器 地 址 栏 中 输入 “http://127.0.0.1/hello.php”， 可 以 看 到 如 图 5-5 所 示 的 效果 。 


5.4 PHP 语法 基础 


5.4.1 常量 


常量 就 是 在 脚本 执行 期 间 其 值 不 会 改变 的 量 。 通 常 都 会 给 常量 起 一 个 用 于 识别 的 
名 称 ， 这 个 识别 名 称 就 是 标识 符 。 标 识 符 通 常 都 要 遵循 PHP 的 命名 规范 ， 即 以 字母 或 
下 画 线 开 头 ， 后 面 跟 任何 字母 、 数 字 或 下 画 线 。 

常量 默认 大 小 写 敏感 ， 但 按照 惯例 ， 常 量 名 均 采 用 大 写 形式 。 常 量 的 作用 域 是 全 
局 的 ， 不 管 常量 是 在 哪个 作用 范围 内 定义 的 ， 均 可 在 程序 的 任何 位 置 进行 调用 。 一 个 
常量 一 旦 被 定义 ， 就 不 能 修改 或 取消 。 定 义 常量 的 语法 格式 如 下 。 
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bool define(string $name,mixed $value,[,bool Scase_insensitive]) 


其 中 ，$name 是 常量 名 ;$value 是 常量 的 值 ， 布尔 型 $case_insensitive 表示 常量 名 是 否 
大 小 写 敏感 ， 若 为 te， 则 表示 变量 名 大 小 写 不 敏感 ， 若 为 false， 则 表示 变量 名 大 小 
写 敏感 ， 默 认为 false。 

下 面 的 代码 演示 了 常量 的 几 种 形式 。 


<?php 

define("DEFAULT PATH","/var/www/"): /定义 常量 ， 大 小 写 敏 感 

define("UPLOADS PATH","/var/www/uploads/",true); // 定 义 常 量 ， 大 小 写 不 敏感 

echo default_path; /1/ 打 印 常 量 ， 未 定义 

echo DEFAULT PATH /1/ 打 印 常 量 ， 输 出 /Var/www/ 

echo UPLOADS _PATH: // 打 印 常量 ， 输 出 
/var/www/uploads/ 

echo uploads_path; /打印 常量 ， 输 出 
/var/www/uploads/ 

?> 


从 上 面 的 示例 可 以 看 到 ， 采 用 大 小 写 不 敏感 的 方式 进行 定义 ， 虽 然 在 引用 时 比较 
方便 ， 但 为 了 和 惯例 靠拢 ， 应 尽量 采用 大 小 写 敏感 的 方式 进行 常量 的 定义 。 


5.4.2 系统 常量 


PHP 提供 了 大 量 的 系统 预定 义 常量 ， 分 为 两 类 : 一 类 是 内 核 预定 义 常 量 ， 一 类 是 
标准 预定 义 常量 。 内 核 预定 义 常量 是 在 PHP 的 内 核 中 进行 定义 的 ， 而 标准 预定 义 常量 
是 在 PHP 中 默认 定义 的 。 大 部 分 常量 是 由 不 同 的 扩展 库 定义 的 ， 只 有 加 载 了 这 些 扩 展 
库 ， 这 些 预 定义 常量 才 可 以 使 用 。 

常量 在 整个 脚本 运行 过 程 中 的 值 是 不 变 的 ， 但 是 有 这 样 几 个 魔术 常量 ， 它 们 的 值 
会 根据 使 用 的 位 置 而 改变 。 例 如 ，_FILE_ 常量 的 值 会 随 它 所 在 的 脚本 文件 的 不 同 而 
不 同 。 而 实际 应 用 中 可 在 脚本 文件 中 直接 使 用 这 些 变 量 。 代 码 如 下 所 示 。 


<?php 

echo "Operation System:" . PHP_OS . "\n"; /1/ 打 印 出 操作 系统 名 称 
echo "PHP version:" . PHP_VERSION . "\n"; 1/ 打印 出 PHP 版 本 
echo "current line:" .LINE "\n"; // 显 示 当 前 行 数 

wea 


对 于 更 多 的 系统 常量 ， 大 家 在 应 用 时 可 自行 查看 相关 文档 ， 这 里 不 做 过 多 介绍 。 
5.4.3 ”变量 


在 PHP 中 , 变量 采用 美元 符号 ($) 加 变量 名 来 表示 。 变量 名 是 大 小 写 敏感 的 , 下 面 
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的 代码 定义 了 两 个 不 同 的 变量 。 


Svar = 'this is a string'; // 定 义 一 个 字符 串 类 型 的 变量 
SVar = 150; // 定 义 一 个 整 型 变量 


其 中 ，var、Var 是 两 个 不 同 的 变量 。 


0 变量 在 使 用 前 无 须 预先 定义 ， 可 以 直接 在 使 用 时 定义 变量 ， 但 是 ; 
在 类 中 的 变量 却 需要 先 定义 再 使 用 。 同 时 在 定义 变量 时 ， 也 可 以 不 用 初始 化 变 ; 
量 ， 未 初始 化 的 变量 具有 其 类 型 的 默认 值 。 


变量 赋值 有 以 下 两 种 。 

第 一 种 : 传 值 赋值 (PHP 默认 的 )。 

第 二 种 : 引用 赋值 ， 指 新 变量 直接 引用 了 原始 变量 。 这 种 方式 赋值 需要 在 要 赋值 
的 变量 前 加 一 个 “&” 符 号 来 实现 。 下 面 的 代码 说 明了 两 种 赋值 方式 的 应 用 。 


<?php 
$a=" 张 三 "; // 采 用 传 值 方式 赋值 
Sb = $a; // 将 $a 赋值 给 Sb 
echo "a:" . $a . "mn"; // 输 出 a: 张 三 
echo "b:" . $b . "\n"; // 输 出 b: 张 三 
$a=" 李 四 "; // 定 义 变量 
echo "a:" . $a."\n"; // 输 出 a: 李 四 
echo "b:" . $b .ny"; // 输 出 b: 张 三 
// 采 用 引用 赋值 的 方式 进行 赋值 
$c = &$b; 
echo "c:" . $e . nn"; // 输 出 c: 张 三 
b= 
echo "a:" . $a."\n"; //a: 李 四 
echo "b:" . $b . "\n"; /b: 王 五 
echo Mon- So. Na, /b: 王 五 
> 


在 上 述 程序 中 ， 采 用 了 传 值 赋值 和 引用 赋值 的 方式 进行 赋值 ， 大 家 对 比 程序 结果 
可 知 ， 采 用 传 值 赋值 时 ， 程 序 第 6 行 改变 变量 $a 的 值 ， 另 一 变量 $b 不 会 发 生变 化 。 而 
采用 引用 赋值 时 ， 程 序 中 改变 变量 gb 的 值 时 ， 变 量 $c 的 值 发 生 了 变化 ， 而 $a 的 值 不 
会 发 生变 化 。 


%/ 证 守 竹下 下 生 下 放下 大 多 


上 例 代码 中 点 号 用 于 连接 字符 囊 。 
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5.4.4 PHP 数据 类 型 


在 PHP 中 支持 8 种 原始 数据 类 型 :其 中 包括 4 种 标量 类 型 ,分 别 为 布尔 型 (boolean)、 
整 型 (integer)、 浮 点 型 (float) 和 字符 串 类 型 (string); 两 种 复合 类 型 ， 分 别 为 数组 (array) 
和 对 象 (object); 两 种 特殊 类 型 ， 分 别 为 资源 (resource) 和 空 值 (null)。 

PHP 是 一 种 弱 类 型 的 语言 loosely typed language， 也 译作 “宽松 类 型 ”或 “松散 
类 型 ” )。 在 定义 时 ， 无 须 事先 声明 其 数据 类 型 ，PHP 会 自动 在 运行 时 将 其 转换 成 相应 
的 数据 类 型 。 

PHP 的 这 种 自动 设置 数据 类 型 的 特性 给 程序 员 提供 了 相当 大 的 便利 , 但 是 也 存 
在 一 定 的 隐患 。 很 可 能 PHP 在 运行 时 设置 的 数据 类 型 与 程序 员 在 开发 时 所 设想 的 
数据 类 型 完全 相反 。 因 此 ，PHP 提供 了 强制 类 型 转换 来 弥补 这 一 问题 。 例 如 ， 程 序 
员 在 开发 时 可 以 直接 使 用 settype 等 函数 强制 转换 为 程序 所 需要 的 数据 类 型 。 参 考 
表 5-1。 


表 5-1 


类 型 数据 类 型 


(boolean) 


其 值 可 以 用 十 进 制 、 八 进 制 和 十 六 进 制 来 表示 ， 在 其 前 加 “+” 
整 型 (integer) 和 “_” 表 示 正 负 


浮 点 数 的 字 长 和 平台 无 关 ， 通 常 浮 点 数 的 精度 存在 一 些 问题 ， 在 
标量 数据 | 学 点 型 (float) 应 用 浮 点 数 时 ， 尽 量 不 要 将 一 个 很 大 的 数 和 一 个 很 小 的 数 相 加 减 ， 
类 型 


否则 很 小 的 数 会 被 忽略 


车 想 在 输出 字符 串 的 同时 输出 单 引号 ， 需 要 使 用 转 义 序列 符 
(进行 转 义 ( 转 义 符 用 于 将 不 能 在 字符 串 中 直接 输出 的 字符 
字符 申 型 进行 转 义 )。 若 想 输 出 反 斜 杠 ， 需 要 使 用 \。 若 想 将 两 个 字 
(string) 符 串 或 多 个 连接 起 来 ， 可 使 用 点 () 运 算 符 来 连接 。 采 用 三 种 


方式 来 表示 : 单 引 号 、 双 引号 和 定 界 符 (<<<EOF EOF， 详 见 
下 面 示例 ) 
复合 数据 
类 型 类 似 于 Java 和 C# 中 的 对 象 
资源 是 一 种 特殊 的 变量 ， 它 保存 着 对 外 部 资源 的 一 个 引用 。 例 
如 ,一 个 网 站 中 的 超 链接 ， 其 实际 的 内 容 为 其 指向 的 如 URL 或 
特殊 数据 者 图 片 资源 。 由 于 资源 类 型 变量 保存 着 打开 文件 、 数据 库 连 接 、 
类 型 Ce 图 形 画 布 区 域 等 诸多 特殊 内 容 ， 是 一 个 个 体 ， 因 而 无 法 将 其 他 
类 型 的 数值 转换 为 资源 类 型 


空 值 (pull) 表示 一 个 没有 值 的 变量 
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5.4.5 PHP 运算 符 和 表达 式 


1. 赋值 运算 符 

赋值 运算 符 是 简单 的 一 种 运算 符 ， 与 Java 和 C# 一 样 使 用 等 号 (=)。 
2. 算术 运算 符 

算术 运算 符 如 表 5-2 所 示 。 


将 $a 与 Sb 相 加 

将 $a 与 $b 相 减 

将 $a 与 $b 相 乘 

将 $a 与 $b 相 除 

将 $a 与 Sb 相 除 ， 取 余数 
将 $a 自 加 1 

将 $a 自 减 1 


3. 字符 串 运 算 符 
字符 串 运算 符 如 表 5-3 所 示 。 


将 Sa 和 $b 连接 起 来 
连接 字符 串 $a 与 Sb， 并 赋值 给 $a 


下 面 看 一 个 字符 串 连 接 的 示例 。 


<?php 

$a= 12; 

$b = 

str = "我 被 连接 起 来 了 '; 

/字符 串 连接 

echo $a.$b.$str; 

// 三 元 运算 符 

echo $a > $b? "天 外 飞 仙 " : " 降 龙 十 八 掌 "; ?> 
4. 比较 运算 符 


比较 运算 符 如 表 5-4 所 示 。 
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功 能 
$a 等 于 Sb， 就 返回 true 
$a 等 于 $b， 且 它们 的 类 型 相等 ， 就 返 
回 tme 
$al=$Sb $a<>$b $a 不 等 于 Sb， 就 返回 true 


Sa 不 等 于 sb， 或 者 它们 的 类 型 不 相 


Sa 小 于 sb， 训 返回 me 
Sa 大 sb， 训 返回 we 
Sa 小 等 了 sb， 就 返回 me 
Sa 大 于 等 下 sb， 就 返回 tne 


逻辑 与 $a and $b,$a && $b 如 果 $a 和 $b 都 为 tue， 就 返回 true 
Sa 或 sb 任 一 为 rve， 返 回 te 
如 果 $a 和 $b 有 且 仅 有 一 个 为 tue， 


逻辑 异 或 $a xor $b 就 返回 true 


6. 位 运算 符 


PHP 中 的 位 运算 符 允 许 对 整 型 数 中 的 指定 位 进行 移动 操作 (注意 :这 里 的 位 是 二 
ee 如 果 运 算 符 左右 参数 都 是 字符 串 ， 则 位 运算 符 操 作 字符 串 的 ASCI 值 。 
运算 符 如 表 5-6 所 示 。 
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功 能 
如 果 $a 和 $b 相对 应 的 位 都 为 1， 则 结果 中 
的 该 位 为 1 
如 果 $a 和 $b 相对 应 的 位 有 一 个 为 1， 则 结 
果 中 的 该 位 为 1 
如 果 $a 和 $b 相对 应 的 位 不 同 ， 则 结果 中 的 
该 位 为 1 
将 $a 中 为 0 的 位 置 改 为 1, 为 1 的 位 置 改 为 
0 


将 $a 中 的 位 向 左 移动 Sb 位 (每 一 次 移动 都 相 
当 于 乘 以 2) 

将 $a 中 的 位 向 右 移动 Sb 位 (每 一 次 移动 都 相 
当 于 除 以 2) 


7. 表达 式 


PHP 中 的 表达 式 可 分 为 赋值 运算 表达 式 、 算 术 运 算 表 达 式 、 字 符 串 运算 表达 式 、 
比较 运算 表达 式 、 罗 辑 运 算 表 达 式 和 位 运算 表达 式 ， 使 用 方法 与 Java 和 C# 中 的 大 同 


小 异 ， 这 里 不 再 袭 述 ， 在 后 面 的 示例 中 会 使 用 到 。 


5.5 _ PHP 流程 控制 


PHP 提供 了 一 组 流程 控制 语句 用 于 实现 程序 执行 顺序 。 这 里 的 条 件 控制 语句 和 循 
环 控制 语句 与 Java 和 C# 中 的 区 别 不 大 ， 此 处 不 再 详细 介绍 语法 基础 ， 只 通过 案例 来 
讲解 用 法 ， 并 对 跳 转 语句 做 单独 介绍 。 

e 条 件 控制 语句 : if 和 switch 


<?php 
// 每 一 行 语句 结束 时 用 分 号 结束 
$a ="Hello LAMP!":; 
$a =" Hello Word!"; 
echo $a; 
iflisset($a)){ // 函 数 isset 用 于 检测 变量 是 否 已 经 被 赋值 ， 返 回 值 为 布尔 类 型 
echo ' 变 量 $a 已 经 被 定义 '; 
jelse{ 
echo "变量 $a 还 未 被 定义 '; 
上 
了 


e 循环 控制 语句 : while、do-while、for 和 foreach 
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e 跳 转 控制 语句 : break、continue 和 retum 


5.6 ” 跳 转 语句 


跳 转 语句 用 于 实现 程序 流程 的 跳 转 ，PHP 提供 了 3 种 跳 转 语 句 : break、continue 
和 return。 


5.6.1 break 跳 转 语句 


break 跳 转 语句 用 于 结束 当前 while、do-while、for、foreach 循环 和 switch 分 支 语 
句 的 执行 。 它 可 接受 一 个 可 选 的 数字 来 决定 跳出 几 层 循环 。 
下 面 的 示例 演示 了 break 语句 的 使 用 。 


<!-- 在 下 面 程序 中 ， 当 变量 为 5 时 ， 则 直接 跳出 外 层 循环 --> 
<?php 
for($a=10 ; $a > 0; $a--){ 
switch($a){ 
case 3: // 当 变量 为 3 时 ， 跳 出 当前 循环 
echo $a . "\n"; 
break 1; 
case 5: // 当 变量 为 5 时， 跳出 两 层 循环 


// 当 变量 为 其 他 值 时 不 执行 任何 操作 


5.6.2 ”continue 跳 转 语句 


continue 语句 用 于 在 循环 结构 中 跳 过 本 次 循环 的 剩余 代码 ， 并 在 条 件 表 达 式 为 true 
时 进行 下 一 次 循环 。 
下 面 的 示例 演示 了 continue 语句 的 使 用 。 


<?php 
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for($a=10 ; $a > 0; $a--){ 
Switch($a){ 
case 3: // 当 变量 为 3 时 ， 跳 出 本 次 循环 ， 进 行 下 一 次 循环 
echo $a . "m'"; 
continue 1; 
case 5: // 当 变量 为 5 时， 跳出 外 层 循环 ， 进 行 下 一 次 循环 
echo $a . "\n"; 
continue 2; 
default: // 当 变量 为 其 他 值 时 不 执行 任何 操作 
break:; 


使 用 时 要 注意 ，break 是 跳出 整个 循环 体 ， 而 continue 则 是 跳 过 本 次 循环 ， 并 继续 
向 下 执行 。 使 用 时 一 定 要 注意 两 者 的 区 别 。 


5.6.3 return 跳 转 语句 


return 语句 用 于 结束 一 个 函数 或 文件 。 如 果 一 个 函数 中 使 用 return 语句 ， 将 结束 
函数 的 执行 ,并 将 其 参数 作为 函数 的 值 返 回 。 但 如 果 是 在 全 局 范围 内 使 用 return 语句 ， 
则 终止 当前 脚本 文件 的 执行 。 如 果 当 前 脚本 文件 是 被 include 或 是 require 的 ， 则 控制 
交 回 调用 文件 。 如 果 脚 本 文件 是 被 include 的 ，retur 语句 的 值 将 被 作为 include 的 返 
回 值 。 下 面 对 文 件 包含 进行 详细 讲解 。 


5.7 文件 包 合 


在 实际 应 用 开发 中 ， 常 常 需要 将 一 些 公 用 的 程序 代码 放 到 一 个 单独 的 文件 中 ， 
而 其 他 文件 在 需要 使 用 这 些 代码 时 只 需要 将 此 单独 的 文件 包含 即 可 ， 这 有 利于 代码 
的 重用 。 


5.7.1 使 用 include 和 include_once 包含 文件 


include 语句 与 include_once 语句 的 文件 包含 功能 和 使 用 方法 一 样 ， 但 存在 一 点 小 
的 差别 。 


1. include 语句 
include 语句 用 于 包含 并 运行 指定 文件 ， 其 语法 格式 如 下 。 


include(string filename) 
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其 中 ，string 表示 要 引用 的 文件 名 的 类 型 为 字符 串 型 ; filename 为 要 包含 的 文件 名 。 找 
寻 所 包含 的 文件 名 是 从 当前 工作 目录 相对 的 include_path 开始 查找 的 ， 然 后 查找 当前 
脚本 文件 所 在 目录 的 相对 include_path。 但 如 果 文 件 名 是 以 (./) 或 者 (../) 开 始 ， 则 只 在 当 
前 工作 目录 相对 的 include_path 下 找寻 。 

当 一 个 文件 被 包含 时 , 该 文件 继承 了 包含 该 文件 所 在 行 的 变量 范围 。 从 该 处 开始 ， 
调用 文件 在 该 处 可 用 的 所 有 变量 在 被 调用 文件 中 均 可 直接 进行 调用 ， 但 是 所 有 在 被 包 
含 文件 中 定义 的 函数 和 类 都 具有 全 局 作用 域 。 

下 面 示例 演示 了 include 语句 实现 文件 的 包含 ， 代 码 如 下 。 


<?php 

Sname =" 李 明博 " ; 

echo "namel:" .$name. "\n"; 
echo "agel:" .$age. "\n"; 
include("inctest.php"); 

echo "name2:" .$name. "\n"; 
echo "age2:" .$age. "\n"; 

?> 


其 中 ，inctestphp 为 被 包含 的 文件 ， 该 文件 代码 如 下 。 


<?php 

echo "namel:" .$name. "\n"; 
Sname = "杰克 "; 

$age= 18; 

echo "name2:" .$name. "\n"; 
echo "age2:" .$age. "\n"; 

?> 


运行 结果 如 下 。 


namel: 李 明博 
agel: 
namel: 李 明博 
name2: 杰 克 
age2:18 
name2: 杰 克 
age2:18 
从 上 面 的 结果 可 以 看 出 ， 在 包含 文件 中 定义 的 变量 Sname 在 被 包含 文件 中 一 样 有 
效 , 同时 在 被 包含 文件 中 对 变量 Sname 进行 再 次 赋值 , 在 包含 文件 中 也 可 直接 进行 引用 ， 
说 明 被 包含 文件 中 定义 的 变量 、 函 数 等 具有 全 局 作用 域 。 
对 于 被 包含 的 文件 ， 还 可 以 使 用 retum 语句 来 终止 它 的 执行 ， 并 返回 调用 文件 ， 
也 可 使 用 return 语句 返回 一 个 值 。 
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注意 
在 被 包含 文件 中 使 用 return 语句 来 返回 值 ， 被 包含 文件 只 能 是 在 本 地 文件 ; 
中 使 用 。 


下 面 在 被 包含 文件 中 采用 retum 语句 返回 值 ， 代 码 如 下 。 


<?php 

$name =" 李 明博 " ; 

echo include("inctest2.php"): 
?> 


其 中 被 包含 的 文件 inctest2.php， 代 码 如 下 。 


<?php 
return $name ." 是 韩国 前 总 统 "; 
?> 


输出 结果 如 下 。 


李 明 博 是 韩国 前 总 统 
在 上 面 的 程序 中 ， 被 包含 文件 中 采用 retum 语句 来 返回 值 ， 如 同 直 接 将 被 包含 文 


件 中 的 语句 写 在 包含 文件 中 一 样 。 
2. include_once 语句 


include_once 语句 在 脚本 执行 期 间 包 含 并 运行 指定 文件 。 它 与 include 语句 非常 类 似 ， 
唯一 区 别 在 于 ， 如 果 该 文件 中 的 代码 已 被 包含 了 ， 则 不 会 被 再 次 包含 。 而 include 语句 就 
会 再 次 包含 ， 这 就 很 容易 出 现 变 量 重新 赋值 、 函 数 重复 定义 等 问题 。 为 避免 出 现 类 似 的 
问题 ， 推 荐 使 用 include_once 语句 。 

include_once 语句 的 返回 值 也 与 include 语句 一 样 。 如 果 文件 已 被 包含 , 则 函数 返回 tue。 

下 面 的 示例 先 看 看 用 include 语句 会 出 现 什么 问题 。 


<?php 

Sname =" 李 明博 "; 

echo include('include3.php"): 
Include(include4 php)): 

?> 


其 中 ，include3.php 被 包含 文件 代码 如 下 。 
<2php 
Teturn $var; 
| 
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return show($name); 
了 > 


include4.php 被 包含 文件 代码 如 下 。 
<2php 
include(include3.php); 


echo show($name); 
?> 


上 面 的 程序 在 运行 时 会 报 致命 错误 , 如 果 将 include 语句 改 为 include_once 语句 就 
不 会 了 。 因 为 使 用 include_once 语句 ， 被 包含 过 的 文件 就 不 会 再 被 包含 。 


5.7.2 使 用 require 和 require_once 包含 文件 


require 语句 包含 并 执行 被 包含 的 文件 ， 它 的 用 法 和 功能 与 include 语句 一 样 。 
require_once 语句 与 require 语句 一 样 , 唯一 的 区 别 就 在 于 如 果 文 件 中 的 代码 已 被 包含 ， 
则 不 会 被 再 次 包含 进去 。require 与 include 语句 的 区 别 在 于 以 下 两 点 。 


1. 包含 机 制 


require 语句 在 包含 文件 时 ， 不 管 该 包含 语句 是 否 被 执行 ， 被 包含 文件 都 将 被 包含 
进去 ; 而 对 于 使 用 include 语句 包含 文件 时 ， 如 果 该 包含 语句 没有 被 执行 ， 那 被 包含 文 
件 不 会 被 包含 进去 。 

下 面 的 代码 采用 了 require 语句 来 包含 文件 。 


<?php 

$a= 10; 

if($a < O){ // 如 果 变 量 $a 小 于 0， 则 包含 文件 inctest3.php 
Tequire('inctest3.php'); 

} 

?> 


上 面 的 代码 没有 任何 输出 , 但 由 于 require 语句 的 包含 机 制 , 不 管 这 条件 表达 式 为 
true 还 是 false， 被 包含 文件 inctest3.php 都 将 被 包含 进来 。 
下 面 的 代码 采用 了 include 语句 来 包含 文件 。 


<?php 

$a= 10; 

if($a < 0){ /如 果 变 量 $a 小 于 0， 则 包含 文件 inctest3.php 
include('inctest3.php’); 

} 

ye 


上 面 的 代码 中 ， 只 有 当 变 量 $a 小 于 0 时 ， 才 会 将 文件 inctest3.php 包含 进去 。 
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2. 错误 处 理 


在 包含 文件 时 ， 如 果 找 不 到 被 包含 的 文件 ，require 语句 会 抛 出 一 个 致命 错误 ， 并 
停止 脚本 执行 ， 而 include 语句 则 只 会 抛 出 一 个 警告 信息 ， 并 继续 执行 其 后 的 脚本 。 
下 面 用 require 语句 包含 一 个 不 存在 的 文件 ， 代 码 如 下 。 


<?php 

$a = 10; 

if($a> 0){ /如 果 变 量 $a 小 于 0， 则 包含 文件 inctest3.php 
Tequire(inctest3.php); 

} 

echo "a"; 

Ms 


| 三 址 @) [各 hp//127 oo ytest php 


Warning: require(1.php) [function. require]: failed to open stream: No such 
file or directory in C:\wamp\www\test.php on line 


Fatal error: require() [function. require]: Failed opening required 
"1.php’ (include_path=’. ;C:\php5\pear’) in C:\wamp\mew\test. php on line 2 


下 面 用 include 语句 包含 一 个 不 存在 的 文件 。 代 码 如 下 所 示 。 


<?php 
include(inctest3.php); 
echo "a"; 

?> 


运行 结果 如 图 5-7 所 示 。 


oS DOOR 


这 收 荐 天 Bt /localhost/a php 


i 页 面 四 ” 实 全 加 - re a 


= 
Warning- et te function include]: failed to open stream: No such file or directory in 
D:\WWWa.php on line 


Warning: include() onction in de ee bie oz 吉首 .php' for inclusion 
(oe >_path=.:C:\php\pear’) in D:\WWW\a.php on 
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【单元 小 结 】 


PHP 简介 

PHP 开发 环境 及 配置 

PHP 常用 的 编辑 器 

PHP 的 基本 语法 以 及 控制 流程 
PHP 的 函数 格式 


【单元 自 测 】 


1. PHP 是 一 种 ( 。”) 脚 本 语言 ,基于 ( 。 ”) 引 擎 .PHP 最 常 被 用 来 开发 动态 的 ( 
内 容 ， 此 外 ， 它 同样 还 可 用 来 生成 (。”) (以 及 其 他 ) 文 档 。 
A. 动态 ，PHP， 数 据 库 ，HTML 
B. 媒 入 式 ，Zend，HTML，XML 
C. 基于 Perl 的 ，PHP，Web， 静 态 
D. 嵌入 式 ，Zend，Docbook 文档 ，MySQL 
.以 下 哪 种 标签 不 是 PHP 起 始 /结束 符 ? ( ) 


. 以 下 代码 哪个 不 符合 PHP 语法 ? (  ) 
A.$_10 
B. ${"MyVar"} 
C. &$something 
D. $10_somethings 
. 运行 以 下 代码 将 显示 什么 ? (。 ) 


<?php 

define(myvalue, "10"); 
Smyarray[10] = "Dog"; 
Smyarray[] = "Human"; 
Smyarray['myvalue'’] = "Cat": 
Smyarray["Dog"] = "Cat"; 
Pprint "The value is: "; 

Print $myarray[myvalue]."\n":; 
?> 


A. The Value is: Dog 
B. The Value is: Cat 
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C. The Value is: Human 
D. The Value is: 10 
5. print() 和 echo() 有 什么 区 别 ?(  ) 
A. print() 能 作为 表达 式 的 一 部 分 ，echo0) 不 能 
B. echo0 能 作为 表达 式 的 一 部 分 ，printO 不 能 
C. echo0 能 在 CLI( 命 令 行 ) 版 本 的 PHP 中 使 用 ，print0 不 能 
D. printO) 能 在 CLI( 命 令 行 ) 版 本 的 PHP 中 使 用 ，echo0) 不 能 


【上 机 实战 】 


上 机 目标 
PHP 开发 环境 及 配置 
PHP 常用 的 编辑 器 


PHP 的 基本 语法 以 及 控制 流程 
PHP 的 函数 格式 


上 机 练习 


人 第 一 阶段 多 


练习 : 在 PHP 页 面 中 采用 switch 语句 进行 判断 。 


【问题 描述 】 


在 PHP 页 面 中 ， 进行 多 个 条 件 判 断 时 ， 虽然 可 以 使 用 felse 语句 进行 处 理 ， 但 若 
采用 switch 语句 进行 判断 将 显得 更 有 效率 。 


【问题 分 析 】 


(1) 在 PHP 页 面 中 根据 变量 url 的 值 进行 判断 ， 选 择 合适 的 程序 分 支 执行 。 
(2) 使 用 “Zend Studio for Eclipse” 新 建 PHP 项 目 。 


【参考 步骤 】 
(1) 打开 “Zend Studio for Eclipse”， 软 件 界 面 如 图 5-8 所 示 。 
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图 5-8 


(2) 新 建 一 个 PHP 项 目 Test， 页 面 代码 如 下 。 


<?php 
Surl = "www.phpcoding.cn"; 
switch($urD){ 
Case "WWW": 
echo "这 是 网 站 主机 "; 
break:; 


case "phpcoding": 
echo "这 是 网 站 名 称 主体 "; 
break; 


Case "co": 
echo "这 表示 国内 域名 "; 
break: 


default: 
echo "这 就 是 全 部 的 网 址 :$url"; 
break:; 


} 
?> 


G) 运行 效果 如 图 5-9 所 示 。 
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练习 : 在 PHP 页 面 中 练习 进行 终止 、 跳 出 循环 和 结束 程序 的 执行 。 


【问题 描述 】 


在 PHP 页 面 中 使 用 break、continue、exit 来 进行 程序 的 终止 、 跳 出 循环 和 结束 程 
序 的 执行 。 


【参考 代码 】 


<?php 

//break 语句 终止 循环 : 

Snum = 1; 

while($Snum < 5){ 
echo "While Break: " . Snum . "<br />"; 
break; 

} 


/continue 语句 直接 跳 到 下 一 次 循环 : 
Snum = 1; 
while(Snum<5){ 
Snumt+; 
continue; 
echo "While Continue: " . Snum ."<br>"; // 因 为 上 面 已 经 直接 转 到 下 一 次 循环 了 , 所 以 不 
会 有 任何 输出 
B 


/exit 语句 停止 脚本 执行 : 
$num = 1; 
这 $num =—D{ 


echo "Exit out: 变 量 num 的 值 :$num"; 
exit; 
echo "这 条 语句 不 会 输出 "; 
了 
?> 
运行 结果 如 图 5-10 所 示 。 


ee [localhost/Test/bre: 一 一 Microsoft Internet Explorer 3 
Cs B I es 5 
地 址 @) | 图 http://ocahost/Testjoreak.php 直 | 四 条 


While Break: 1 
Exit out: 变 量 num 的 值 :1 


加 
了 列宁 [ [和 [ nvenet 


【拓展 作业 】 


1. 写 一 段 代码 实现 如 下 功能 : 判断 给 定 变量 是 否 大 于 5， 并 在 其 大 于 5 时 输出 相 
应 的 结果 。 
2. 写 一 段 程序 ， 使 用 while 循环 输出 1 一 20 之 间 的 数 。 
3. PHP 代表 什么 意思 ? ( ””) (选择 一 项 ) 
A. Hypertext Preprocessor( 超 文本 预 处 理 器 ) 
B. Hyperlink Preprocessor( 超 链接 预 处 理 器 ) 
C. Personal Homepage (个 人 主页 空间 ) 
D. Page Hyperlink Page( 超 级 链接 页 面 ) 
. 以 下 PHP 标记 哪个 是 错误 的 ? (  ) (选择 一 项 ) 
A. <? ?> 
B. <?PHP?> 
C.<% %> 
D .<SCRIPT language="php">...</script> 
5. 下 边 哪个 变量 是 非法 的 ? (””) (选择 一 项 ) 
A.S$ 10 
B. ${"MyVar"} 
C. &$something 
D. $aVaR 
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6. 选择 下 面 程序 运行 的 结果 ( 。”)。( 选 择 一 项 ) 
<?php $a=l1;$a=++;$c=&$a;Sb=$ctt;echo "\$a=$a <br> \Sb=$b <br> \Sc=$e";?> 


A. $a=3 $b=2 $c=3 

B; $a=3 $b=2 $c=2 

C.a=2 $b=2 $c=3 

D. $a=2 

7. PHP 中 的 标量 数据 类 型 有 如 下 几 种 ,布尔 型 (boolean)、 整 型 (integer)、 浮 点 型 (float)、 
字符 串 型 (string)， 其 中 布尔 型 (boolean) 的 返回 值 是 (  )。( 选 择 两 项 ) 
Bl 
D. False 


PHP 数组 和 字符 外 


时 课程 目标 


P 掌握 PHP 数组 
P 掌握 字符 串 的 常用 操作 
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多 > 个 


PHP 提供 了 数组 数据 类 型 ， 还 提供 了 与 数组 操作 有 关 的 大 量 行为 和 函数 。PHP 的 
数组 实际 上 就 是 一 组 变量 的 集合 ， 每 个 变量 都 叫 作 数组 的 一 个 元 素 。 对 每 一 个 元 素 的 
引用 都 有 一 个 索引 值 来 唯一 确定 ， 这 个 索引 值 就 是 数组 的 键 ， 而 这 个 变量 就 是 键 所 对 
应 的 值 。 它 可 以 是 一 维 的 ， 也 可 以 是 多 维 的 。 多 维 数组 可 以 理解 为 以 数组 为 元 素 的 数 
组 。 例 如 ， 二 维 数组 就 可 以 理解 为 其 每 一 个 元 素 都 是 一 个 一 维 数组 。 


6.1 一 维 数组 和 多 维 数 组 


1. 一 维 数组 


一 维 数组 就 是 由 具有 相同 数据 类 型 的 数据 组 成 的 一 个 表 。 它 由 一 组 键 与 值 对 应 的 
数据 组 成 ， 如 表 6-1 所 示 。 


2. 多 维 数组 


多 维 数组 就 是 以 数组 为 元 素 的 一 种 特殊 数组 。 最 简单 的 多 维 数组 就 是 二 维 数组 。 
实际 上 ， 二 维 数组 就 是 以 一 维 数组 为 数组 元 素 的 一 种 数组 。 例 如 ， 一 个 3*4 的 二 维 数 
组 可 以 看 作 由 3 个 数组 元 素 为 4 个 的 一 维 数组 构成 ， 如 表 6-2 所 示 。 


从 表 中 可 以 看 出 ， 每 一 行 的 数组 元 素 为 4 个 的 1 个 一 维 数组 ， 而 这 具有 4 个 元 素 
的 3 个 一 维 数组 又 组 成 了 一 个 新 的 数组 一 一 二 维 数组 。 

与 一 维 数组 和 二 维 数组 类 似 , N 维 数组 可 以 看 成 是 多 个 N-1 维 数组 组 成 的 一 个 
数组 。 
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6.2 数组 的 常用 操作 


数组 的 常用 操作 包括 数组 的 创建 与 调用 、 数 组 元 素 的 更 新 与 遍历 等 ， 下 面 将 对 这 
些 操 作 一 一 说 明 。 


6.2.1 数组 的 显示 创建 和 非 显示 创建 


1. 显示 创建 
显示 创建 是 指 直接 应 用 数组 函数 的 方式 来 创建 数组 ， 语 法 格式 如 下 。 
Array([mixed]) 


其 中 ，Array 是 PHP 创建 数组 的 关键 字 ; [表示 参数 是 可 选 的 ， 参 数 mixed 接受 一 定数 
量 用 逗号 分 隔 的 key=>value 参数 对 。 其 中 key 可 以 是 integer 或 string 类 型 ，value 可 以 
是 任何 值 。 如 果 省 略 了 索引 ， 会 自动 产生 从 0 开始 的 整数 索引 。 如 果 索 引 是 整数 ， 则 下 
个 索引 是 当前 最 大 索引 值 加 1。 

下 面 创建 一 个 具有 3 个 元 素 的 一 维 数组 ， 代 码 如 下 。 


<?php 

// 创 建 一 个 名 为 myarr 的 一 维 数组 

$myarr = array(0=>" 张 三 丰 ",1=>15,2=>" 武 汉 "); 

echo "姓名 : " .$myarr[0]."\n"; // 输 出 "姓名 : 张三丰" 

echo "年 龄 : " .$myarr[1]."\n"; // 输 出" 年龄 : 15" 

echo "地 址 : " .Smyarr[2]."\n"; // 输 出 "地 址 : 武汉 " 

> 

这 里 采用 直接 构造 数组 值 对 的 方式 创建 数组 。 这 里 的 0、1、2 就 是 数组 的 键 ， 张 

三 丰 、15、 武 汉 就 是 数组 的 键 所 对 应 的 值 。 


2. 非 显示 创建 
非 显示 创建 数组 是 指 通过 数组 名 加 方 括号 的 方式 来 创建 数组 ， 其 语法 格式 如 下 。 
array[index] 
其 中 ，index 为 指定 的 键 名 。 也 可 省 略 键 名 ， 此 时 只 需要 在 数组 名 后 面 加 上 一 对 空 的 方 
括号 [] 即 可 ， 如 果 数 组 名 不 存在 ， 则 将 新 建 一 个 数组 。 如 果 给 出 方 括号 但 没有 指定 键 
名 ， 则 取 当 前 最 大 整数 索引 值 ， 新 的 键 名 将 是 该 索引 值 加 1。 如 果 当 前 没有 整数 索引 ， 


则 键 名 将 为 0。 如果 指定 的 键 名 已 经 有 值 了 ， 该 值 将 被 覆盖 。 
下 面 的 代码 将 演示 非 显 示 方 式 创建 一 个 一 维 数 组 。 


<?php 
$myarr = array(0=>" 张 三 丰 ",1=>15,2=>" 武 汉 "); 
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echo "姓名 : " .Smyarr[0]."\n"; 

echo "年 龄 : " .Smyar[1]."\n"; 

echo "地 址 : " .$myarr[2]."\n"; 

$myarr[3]= "2011 年 12 月 15 日 "; /采用 非 显 示 方 式 向 数组 myarr 中 添加 元 素 

Print _r($myarr: // 显 示 整 个 数组 

Smyar[] = "该 生成 绩优 秀 "; // 非 显示 方式 向 数组 添加 元 素 , 这 里 将 自动 产生 索引 值 
Print_r($myarrn: // 输 出 整个 数组 

@arr2[] = "中 国人 民 "; // 向 数组 中 添加 元 素 ， 这 里 将 自动 创建 一 个 新 数组 
print_r(@arr?2) // 输 出 整个 数组 

Tx 


上 面 的 代码 第 8 行 ， 采 用 数组 名 加 方 括号 的 方式 添加 元 素 ， 由 于 省 略 了 索引 值 ， 
自动 取 当 前 最 大 索引 值 3 加 1 作为 当前 索引 值 。 第 10 行 ， 数 组 arr2 并 不 存在 ， 自 动 
创建 数组 arr2， 并 以 整数 索引 值 0 作为 当前 索引 值 。 


6.2.2 ”数组 的 调用 与 删除 


1. 数组 的 调用 
数组 调用 语法 与 Java 和 C# 一样 ， 语 法 如 下 。 


array[index] 
/其 中 array 是 数组 名 ，index 是 索引 值 


在 前 面 的 示例 中 ， 采 用 通过 列举 数组 每 一 个 元 素 的 方式 来 输出 整个 数组 。 这 种 方 
式 在 数组 元 素 较 少时 可 以 使 用 ， 但 是 如 果 数 组 中 的 元 素 过 多 ， 这 样 就 过 于 麻烦 ， 且 增 
加 了 工作 量 。 通 常 在 程序 中 采用 foreach 来 遍历 输出 整个 数组 ， 关 于 foreach 的 用 法 ， 
后 面 会 做 详细 介绍 。 

对 于 在 开发 时 进行 的 调试 ， 更 多 采用 print r 和 var_dump 方式 来 直接 输出 整个 数组 。 

下 面 是 采用 print r 和 var_dump 方式 来 输出 数组 的 示例 ， 代 码 如 下 。 


<?php 

/下 面 创 建 一 个 名 为 myarr 的 二 维 数组 

Smyarr = array(0 => array(0=>" 张 三 丰 ",1=>75,2=>" 武 汉 "), 1=> array(0 => " 张 梁山 " ,1 => 
15,2=>" 北 京 ")); 

print_r($myarr): // 采 用 print r 方 式 输出 

var_dump($myarr); // 采 用 var_dump 方式 输出 

?> 


上 面 的 代码 创建 了 一 个 2X3 的 二 维 数组 , 并 采用 两 种 数组 输出 方式 进行 输出 。 这 
两 种 方式 均 可 以 输出 数组 的 结构 和 内 容 ， 但 结果 却 不 大 容易 理解 ， 所 以 通常 只 在 程序 
调试 过 程 中 采用 这 种 方法 。 在 实际 应 用 中 ， 常 常 采用 其 他 方法 进行 数组 内 容 的 输出 ， 
本 单元 会 在 后 面 进行 讲解 。 
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2. 数组 的 删除 


数组 的 删除 是 指 将 整个 数组 或 者 数组 的 某 个 元 素 从 内 存 空间 中 释放 。PHP 提供 了 
unset 函数 来 删除 数组 。 其 语法 格式 如 下 : 


其 中 ，void 表示 没有 返回 值 。 如 果 试 图 获取 返回 值 将 导致 PHP 解析 错误 ，mixed var 
表示 数组 或 者 数组 元 素 。 
下 面 示例 采用 unset 删除 数组 元 素 和 整个 元 素 ， 代 码 如 下 。 


<?php 

$myarr = array(0=>" 张 三 丰 ",1=>75); 

echo "第 一 个 元 素 : ". $myarr[0]; 

echo "第 二 个 元 素 : " . Smyarr[1]; 

unset($myarr[0]): // 删 除 下 标 为 0 的 元 素 
echo "第 一 个 元 素 : " . Smyarr[0]; 

echo "第 二 个 元 素 : " . $myarr[1]; 

unset($myarr); 1/ 删除 整个 数组 
//print_r($myarr); 

?> 


上 面 代码 第 5 行 删除 了 第 一 个 数组 元 素 ， 第 8 行 删除 了 整个 数组 。 如 果 将 第 9 行 
的 注释 取消 ， 将 产生 一 个 PHP 解析 错误 ， 因 为 此 时 数组 myarr 已 经 被 删除 了 。 


6.2.3 ”数组 的 遍历 


PHP 提供 了 一 种 遍历 数组 的 简便 方法 一 -foreach 函数 。foreach 函数 仅 能 用 于 数 
组 ， 当 试图 将 其 用 于 其 他 数据 类 型 或 者 一 个 未 初始 化 的 变量 时 会 产生 错误 。 它 有 两 种 
语法 格式 。 
foreach(array expression as Svalue) 
Statement 


foreach(array_expression as $key => $value) 
statement 


其 中 ， 第 一 种 格式 遍历 给 定 的 array_expression 数组 。 在 每 次 循环 中 ， 当 前 单元 的 值 被 赋 
给 $value， 并 且 数 组 内 部 的 指针 向 前 移 一 位 ， 因 此 下 一 次 循环 将 会 得 到 下 一 个 单元 。 第 二 
种 格式 与 第 一 种 格式 功能 一 样 ， 只 不 过 在 遍历 数组 时 将 当前 单元 的 键 名 赋 给 变量 $key。 
自 PHP5 起 ，foreach 还 可 以 遍历 对 象 ， 这 一 点 将 在 面向 对 象 的 章节 提 到 。 
下 面 的 示例 采用 了 两 种 方式 对 一 个 一 维 数组 进行 遍历 输出 ， 代 码 如 下 。 
<?php 
$exampleArray = armray(" 武 汉 ", "天 津 " "上 海 ", "北京 ", "南京 "); /创建 数组 
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foreach(SexampleArray as $arrValue) { /采用 第 一 种 方式 遍历 数组 

echo "地 址 : SarrValue&nbsp:&nbsp:&nbsp:&nbsp:": // 显 示 元 素 值 

} 

echo "<br>"; 

foreach(SexampleArray as Skey => $value){ // 采 用 第 二 种 方式 遍历 数组 
// 显 示 键 值 和 元 素 什 

echo "地 址 "” .($key+1) . ": ". $value . "&nbsp:;&nbsp;&nbsp;"; 

} 

?> 


地 址 : 武汉 ”地 址 : 天津 ” 地址: 上 海 地址: 北京 地 址 : 南京 
地 址 1, 武汉 ”地 址 2， 天津 ”地 址 3， 上 海 ”地 址 4， 北京 ”地 址 5， 南 京 


第 二 种 方式 可 以 在 遍历 数组 的 同时 ， 读 取 当 前 数组 单元 的 键 名 ， 所 以 大 家 在 实际 
应 用 时 ， 如 果 需 要 键 名 ， 则 使 用 第 三 种 方式 ， 否 则 采用 第 一 种 方式 更 简捷 。 
下 面 示例 采用 翌 套 foreach 遍历 一 个 3*4 的 二 维 数组 ， 代 码 如 下 所 示 。 


<?php 
SexampleArray = array("item1"=>array("a","b","c","d"), 
2=>array("A","B","C","D"), 
array("one","two","three","four")): 
foreach($exampleArray as $key => $arr){ 
echo "$key: &nbsp;é&nbsp;"; 
foreach($arr as $value){ 

echo "$value &nbsp;&nbsp;"; 
} 
echo "<br>"; 
} 
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6.3 数组 的 查找 


6.3.1 顺序 查找 


顺序 查找 是 指 通过 逐一 比较 数组 元 素 的 方式 来 查找 给 定 的 元 素 。 
下 面 的 示例 采用 遍历 数组 的 顺序 查找 方式 进行 查找 ， 代 码 如 下 。 


<?php 
SexampleArray = array("one", "two","three"); /创建 数组 
function lookup($array, Skey){ // 自 定义 顺序 查找 函数 
$cnt = count($array); //count 函数 用 于 返回 数组 中 元 素 的 个 数 
Sfind = false; 
for($i=0; $i<Sent:Sit+ ) { // 遍 历数 组 
if($array[$i] 一 $Skey){ // 如 果 找 到 ， 退 出 
Sfind = true; 
break: 
} 
if ($find) { 
return "在 索引 " .$i. "处 找到 了 !"; 
b 
else{ 
return "未 找到 !"; 
} 
} 
Sresult = lookup($exampleArray, "two"); 
echo $result; 
?> 


程序 运行 结果 如 图 6-3 所 示 。 


上 述 程序 自 定义 顺序 查找 算法 的 函数 ， 如 果 找 到 ， 返 回 该 值 的 索引 值 ， 如 果 没有 
找到 ， 返 回 “ 未 找到 ”。 
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6.3.2 array_search 查找 


PHP 提供 了 array_search 函数 进行 查找 ， 其 语法 格式 如 下 。 
mixed array_search(mixed needle,array haystack [ , bool strict]) 


其 中 ，needle 为 要 查找 的 值 ，mixed 为 needle 的 类 型 ，haystack 为 要 进行 查找 的 数组 ; 
array 为 haystack 的 类 型 ， 即 数组 类 型 ，strict 为 查找 类 型 ， 值 为 bool 类型， 为 true 时 ， 
在 haystack 中 查找 时 还 将 检查 needle 的 类 型 。 函 数 array_search 如 果 查 找 成 功 ， 则 返 
回 该 值 所 在 的 键 名 ， 和 否则 返回 false。 如 果 needle 在 haystack 中 出 现 多 次 ， 则 返回 第 一 
个 匹配 的 键 名 。 

下 面 是 有 关 array_search 函数 的 应 用 ， 代 码 如 下 。 


<?php 

SexampleArray = array(15,"a",30,60,"b","C"); /创建 数组 

echo "15:\t" . array_search(15,SexampleArray) .";":; // 使 用 array_search 函数 进行 搜索 

echo "a:\t" . array_search("a",$exampleArray) . ":"; 

echo "30:\t" . array_search("30",$exampleArray,true) .";"; /此 处 严格 比较 类 型 ， 结果 为 空 ， 
表示 查找 失败 

echo "60:\t" . array_search(60,$exampleArray,false) . ";"; 

echo "30:\t" . array_search(30,$exampleArray,true) . ";"; /此 处 也 严格 比较 类 型 

echo "b:\t" . array_search("b",$exampleArray,true) . ":"; 

echo "b:\t" . array_search("b",$exampleArray,false) . ";"; 

echo "c:\t" . array_search("c",SexampleArray) . ":"; /此 处 结果 为 空 , 在 查找 时 会 比 
较 大 小 写 

echo "C:\t" . array_search("C",SexampleArray,true) . ":": 

echo "C:\t" . array_search("C",$exampleArray,false) . ":"; 

?> 


程序 运行 结果 如 图 6-4 所 示 。 
http://127.0.0.1/2.8.9 


15: 0;a: 1:30: ;60: 3;30: 2;b: 4;b fF: 5:C: §; 


"| 
EE I 


6.4 数组 的 排序 


在 实际 应 用 中 ， 经 常会 遇 到 将 一 组 数据 进行 排序 的 情况 。PHP 对 于 数组 的 排序 提供 
了 多 种 函数 。 
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6.4.1 递增 排序 


递增 排序 是 指 将 数组 元 素 按 升 序 进 行 重新 排列 。PHP 提供 了 sort 函数 对 数组 进行 
递增 排序 。 其 语法 如 下 。 


bool sort (array &array [int sort flags]) 


其 中 ， 函 数 排序 成 功 返 回 布尔 值 tue， 失 败 返 回 布尔 值 false。 函 数 排序 结束 时 数组 单 
元 将 被 从 最 低 到 最 高 重新 安排 。 参 数 array 为 要 排序 的 数组 ;参数 sort_flags 为 排序 类 
型 标记 ， 其 值 的 变化 将 改变 排序 行为 。Sort_flags 的 取 值 有 以 下 几 种 。 

e SORT REGULAR: 正常 比较 单元 (不 改变 类 型 )。 

e SORT_NUMERIC: 单元 被 作为 数字 来 比较 。 

e SORT_STRING: 单元 被 作为 字符 串 来 比较 。 

e SORT LOCALE _STRING: 根据 当前 的 locale 设置 把 单元 当 作 字 符 串 比较 。 

以 下 示例 是 对 sort 函数 的 应 用 ， 代 码 如 下 。 


<?php 

$stringArray = array(2=>"red",5=>"blue",4=>"yellow"): 
sort($stringArray); /对 字符 串 数组 进行 递增 排序 
Print_r($stringArray); 

echo "<br>"; 


SnumArray = 15,2,4,59):// 对 整数 数组 进行 递增 排序 


$Array = array(15.2,4.59);// 对 整 型 数组 按 字符 串 进行 排序 
sort($Array,SORT_STRING): 

Print_r($Array); 

?> 


结果 如 图 6-5 所 示 。 


Array ( [0] => blue [1] => red [2] => yellow ) 
Array ( [0] => 2 [1] => ¢ [2] => 15 [3] => 59 ] 
Array ( [0] => 15 [1] => 2 [2] => 4 [3] =>59 ) 


ed lle tl i i 


使 用 sort 函数 在 对 含有 混合 类 型 值 的 数组 排序 时 要 小 心 ， 因 为 可 能 会 产生 j 
不 可 预知 的 结果 。 
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6.4.2 ”递减 排序 
PHP 提供 了 rsort 函数 对 数组 进行 递减 排序 ， 其 语法 格式 如 下 。 


bool rsort(array  &array [ , int sort flags]) 


其 中 ， 参 数 的 含义 同上 面 的 sort 函数 。 
下 面 采用 rsort 函数 重 写 上 面 的 示例 ， 代 码 如 下 。 


<?php 

S$stringArray = array(2=>"red",5=>"blue",4=>"yellow"): 
Tsort($stringArray); /对 字符 串 数组 进行 递减 排序 
Print_r($stringArray); 

echo "<br>"; 


SnumArray = array(15,2.4,59);// 对 整数 数组 进行 递减 排序 
Tsort($numArray): 

Print r($numArray): 

echo "<br>"; 


$Array = array(15,2,4.59):// 对 整 型 数组 按 字 符 串 进行 排序 
rsort($Array,SORT_STRING); 
Print r($Array); 


2> 
运行 结果 如 图 6-6 


Array ( [0] =》yellow [1] => red [2] => blue ) 
Array ( [0] => 59 [1] => 15 [2] => 4 [3] => 2 ) 
Array ( [0] => 59 [1] => 4 [2] => 2 [3] => 15 ) 


6.5 字符 的 显示 与 格式 化 


在 PHP 的 原始 数据 类 型 中 ， 有 一 种 字符 串 类 型 。 字 符 串 也 就 是 由 一 个 个 的 字符 组 
合 而 成 的 。PHP 的 字符 包括 以 下 4 种 。 

e 数字 : 如 1、2、3 等 。 

@ 字母 : 如 a、b、c 等 。 

e 特殊 字符 ， 如 @、#、$ 等 。 
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e 格式 字符 : 如 nn、\r、\t 等 。 
字符 的 显示 在 实际 应 用 中 经 常用 到 ， 特 别 是 在 项 目 开发 期 间 ， 会 直接 输出 变量 的 
中 间 结 果 进 行 调试 。 下 面 将 对 字符 的 显示 和 格式 化 输出 进行 系统 介绍 。 


6.5.1 字符 的 显示 


在 PHP 中， 字符 的 显示 主要 有 两 种 方式 ， 一 是 采用 print， 一 是 采用 echo。 
1. print 函数 
printO 函 数 输出 一 个 或 多 个 字符 串 。 
语法 代码 如 下 。 
print (strings) 
其 中 ， 各 参数 说 明 如 表 6-3 所 示 。 
表 6-3 


参 数 描 述 
strings 必需 。 发 送 到 输出 的 一 个 或 多 个 字符 串 


党 


printO) 函 数 实际 上 不 是 函数 ， 所 以 用 户 不 必 对 它 使 用 括号 。 Print0 西 交 和 旬 ; ' 
于 echo()。 > 


下 面 的 示例 说 明 print 的 用 法 ， 代 码 如 下 。 


<?php 

$str = "Who's John Adams?"; 
Print $str; 

Print "<br />"; 

print $str."<br />I dom't knowln: 
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再 看 一 个 示例 ， 代 码 如 下 。 


运 


<?php 

$color = "red"; 

Print "Roses are $color"; /这 里 变量 名 使 用 双 引 号 ， 将 输出 变量 gcolor 的 值 

print "<br />"; 

Print 'Roses are $color; /这 里 使 用 单 引 号 ， 不 会 输出 变量 的 值 ， 而 是 输出 变量 的 名 字 
27> 


行 结果 如 图 6-8 所 示 。 


Roses are red 
Roses are $color 


2. echo 函数 


echo() 函 数 输出 一 个 或 多 个 字符 串 。 
语法 代码 如 下 。 


echo(strings) 


其 中 ， 各 参数 说 明 如 表 6-4 所 示 。 


表 6-4 
描述 
需 。 一 个 或 多 个 要 发 送 到 输出 的 字符 串 


echo0) 实 际 上 不 是 一 个 函数 ， 因 此 用 户 无 须 对 其 使 用 括号 。 不 过 ， 如 果 希 望 向 ! 
那么 使 用 括号 会 发 生 解析 错误 。echo0) 函 数 比 a 


下 面 示例 说 明 echo 函数 的 用 法 ， 代 码 如 下 。 


<?php 

$str = "Who's John Adams? "; 
echo $str; 

echo "<br />"; 

echo $str. "<br />I don’t know! "; 
?> 
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运行 效果 如 图 6-9 所 示 。 


下 面 再 看 一 个 示例 ， 代 码 如 下 。 
<?php 
Echo'This','string','was','made','with multiple parameters'; 
?> 


运行 结果 如 图 6-10 所 示 。 


1/ 0 php — Nie aternet 
文件 中 ”编辑 人 查看 WD 收藏 和 ) 工具 CD) 帮助 op 


ma E50 wa TD Wi 
Ey 站 EO) Pe Wenn ©|B'3 oo- -Hu 
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1. 去 除 首尾 空格 或 指定 字符 
在 应 用 中 ， 字 符 串 常常 会 包含 有 空格 ， 在 输出 时 常常 会 将 其 空格 去 掉 ， 通 常 采用 


的 方式 是 使 用 trim 函数 。 
e Trim() 函 数 从 字符 串 的 两 端 删 除 空白 字符 和 其 他 预定 义 字符 。 


语法 如 下 。 
trim(string,charlist) 
其 中 ， 参 数 说 明 如 表 6-5 所 示 。 


必需 。 规 定 要 检查 的 字符 串 
可 选 。 规 定 要 转换 的 字符 串 。 如 果 省 略 该 参数 ， 则 删除 以 下 
所 有 字符 : 
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"0"—NULL 
"\t"—tab 

"m" 一 new line 
"x0B" 一 纵向 列表 符 
号" 一 回 车 

mn "一 普通 空白 字符 


运行 效果 如 图 6-11 所 示 。 


Without trim: Hello World! 
With trim: Hello World! 


从 正中 可 能 看 不 到 去 掉 空格 的 效果 , 大 家 可 以 查看 源 文件 看 看 它 的 HTML 效果 。 


<html> 
<body> 
Without trim: Hello World! & <br/>With trim: Hello World!<body> 
<html> 


2. 大 小 写 转 换 


在 应 用 中 ,经 常会 出 现 大 小 写 不 统一 的 情况 , 通常 采用 strtolower 函数 和 strtoupper 
函数 分 别 将 其 全 部 转换 为 小 写 或 大 写 。 语 法 格式 如 下 : 


String strtolower(string $str)// 转 换 为 小 写 
String strtoupper(string $str)// 转 换 为 大 写 


其 中 ， 参 数 $str 为 要 转换 的 字符 串 。 
下 面 的 示例 演示 了 这 两 个 函数 的 使 用 ， 代 码 如 下 。 


<?php 

$str = "HELLO world"; 

$lowerStr = strtolower($str); ” // 将 字符 串 转换 为 小 写 

SupperStr = strtoupper($str); /将 字符 串 转换 为 大 写 

var_dump($lowerStD; /var dump 函数 显示 关于 一 个 或 多 个 表达 式 的 结构 信息 , 包括 表达 式 的 
类 型 与 值 

echo "<br>"; 
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var_dump($upperStr); 


了 > 


运行 效果 如 图 6-12 所 示 。 


ED 


string(11) “hello world” 
string(11) “HELLO WORLD 


3. 使 用 printf 格式 化 输出 
在 PHP 中 使 用 printf 函数 进行 格式 化 操作 ， 其 语法 格式 如 下 : 


Printf(format,argl,arg2,arg++) 
其 中 ， 参 数 的 含义 如 表 6-6 所 示 。 


。 规 定 字符 串 以 及 如 何 格式 化 其 中 的 变量 
需 。 规 定 插 到 格式 化 字符 串 中 第 一 个 % 符 号 处 的 参数 

。 规 定 插 到 格式 化 字符 串 中 第 二 个 % 符 号 处 的 参数 
选 。 规 定 插 到 格式 化 字符 串 中 第 三 、 四 等 % 符 号 处 的 参数 


a 
' argl1、arg2、Arg++ 等 参数 将 插入 到 主 字 符 串 中 的 百 分 号 (%) 符 号 处 。 | 
; 数 是 逐步 执行 的 .在 第 一 个 % 符 号 中 ,插入 argl, 在 第 二 个 % 符 号 处 ,插入 arg2，; 


输出 格式 一 般 采 用 符号 “%” 加 特定 格式 化 字符 的 形式 。 该 函数 返回 一 个 按 输出 
格式 进行 格式 化 后 的 字符 串 。 格 式 化 字符 串 主 要 包括 以 下 几 种 。 
。 %: 将 直接 输出 百 分 号 ， 不 需要 额外 参数 。 
b: 参数 args 将 被 识别 为 整 型 数 ， 并 且 以 二 进 制 数 的 形式 进行 输出 。 
c: 参数 args 将 被 识别 为 整 型 数 ， 并 且 以 ASCII 码 的 形式 进行 输出 。 
d: 参数 args 将 被 识别 为 整 型 数 ， 并 且 以 有 符号 数 的 形式 进行 输出 。 
e: 参数 args 将 被 识别 为 科学 记 数 法 ， 并 且 以 科学 记 数 法 的 形式 进行 输出 。 
u: 参数 args 将 被 识别 为 整 型 数 ， 并 且 以 无 符号 数 的 形式 进行 输出 。 
f: 参数 args 将 被 识别 为 浮 点 数 ， 并 且 以 浮 点 数 的 形式 进行 输出 。 
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e@ 0: 参数 args 将 被 识别 为 浮 点 数 ， 并 且 以 八进制 的 形式 进行 输出 。 
es: 参数 args 将 被 识别 为 字符 串 ， 并 且 以 字符 串 的 形式 进行 输出 。 
exX: 参数 args 将 被 识别 为 浮 点 数 ， 并 且 以 十 六 进 制 的 形式 进行 输出 ， 其 中 字母 以 小 
写 形式 输出 。 
ee X: 参数 args 将 被 识别 为 浮 点 数 ， 并 且 以 十 六 进 制 的 形式 进行 输出 ， 其 中 字母 以 大 
写 形式 输出 。 
先 看 下 面 的 示例 ， 代 码 如 下 。 
<?php 
$str = "Hello"; 
Snumber = 123; 
printf("%s world. Day number %u",$str,Snumber); // 这 里 用 到 了 %s 和 %u 表示 分 别 以 字符 串 和 
整 型 值 输出 
?> 


运行 效果 如 图 6-13 所 示 。 


Hello world. Day number 123 


图 6-13 


如 果 % 符 号 多 于 arg 参数 ， 则 用 户 必 须 使 用 占 位 符 。 占 位 符 被 插入 % 符 号 之 后 ， 由 
数字 和 “\$” 组 成 。 请 参见 下 面 的 示例 ， 代 码 如 下 。 


<?php 

Snumber = 123; 

// 这 里 % 多 于 arg 参数 ， 所 以 占 位 符 被 插入 % 符 号 之 后 ， 由 数字 和 "S$" 组 成 
printf("With 2 decimals: %1\$.2f<br />With no decimals: %1\$u",$number); 
Ge 


效果 如 图 6-14 所 示 。 


六 
http://127.0.0. 172. 18. php 


运行 
] 


With 2 decimals: 123.00 
With no decimals: 123 
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6.6 字符 串 的 常用 操作 


String 函数 是 PHP 核心 的 组 成 部 分 。 无 须 安装 即 可 使 用 这 些 函数 ， 表 6-7 列 出 了 
string 函数 。 


表 6-7 
支持 该 函数 的 最 早 的 
函数 描 述 BE 段 示 
addcslashes() 在 指定 的 字符 前 添加 反 和 斜 杠 4 
addslashes() 在 指定 的 预定 义 字符 前 添加 反 斜 杠 3 
bin2hex() 把 ASCI 字符 的 字符 串 转换 为 十 六 进 制 值 i 
chopO) rtrim() 的 别名 3 
chrO 从 指定 的 ASCII 值 返回 字符 3 
chunk splitO 把 字符 串 分 割 为 一 连 串 更 小 的 部 分 3 
convert_cyr_string() 把 字符 由 一 种 Cyrillic 字符 转换 成 男 一 种 3 
convert_ uudecode() 对 uuencode 编码 的 字符 串 进行 解码 5 
convert uuencode() 使 用 uuencode 算法 对 字符 串 进行 编码 5 
count_chars() 返回 字符 串 所 用 字符 的 信息 4 
crc320) 计算 一 个 字符 串 的 32-bit CRC 4 
cryptO) 单 向 的 字符 串 加 密 法 (hashing) 3 
echo0) 输出 字符 串 3 
explode() 把 字符 串 打 散 为 数组 3 
fprintf() 把 格式 化 的 字符 串 写 到 指定 的 输出 流 5 
et 返回 翻译 表 4 
下 全 把 希 伯 来 文本 从 右 至 左 的 流转 换 为 从 左 至 3 
右 的 流 
hebrevcO) 同上 ， 同 时 把 (m) 转 为 <br/> 3 
html entity decode() 把 HTML 实体 转换 为 字符 4 
htmlentities() 把 字符 转换 为 HTML 实体 3 
htmlspecialchars_decode() | 把 一 些 预定 义 的 HTML 实体 转换 为 字符 5 
htmlspecialchars() 把 一 些 预 定义 的 字符 转换 为 HIML 实体 3 
implode() 把 数组 元 素 组 合 为 一 个 字符 串 3 
join0 implode() 的 别名 3 
levenshtein() 返回 两 个 字符 串 之 间 的 Levenshtein 距离 3 
localeconv() 返回 包含 本 地 数字 及 货币 信息 格式 的 数组 4 
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( 续 表 ) 
支持 该 函数 的 最 早 的 
函数 描 述 PHP 版 本 
ltrim() 从 : 左 侧 删除 空格 或 其 他 预定 义 字符 3 
md5() 计算 字符 串 的 MDS5 散 列 3 
md5 file() 计算 文件 的 MD5S 散 列 4 
metaphone() 计算 字符 串 的 metaphone 键 4 
money format() 把 字符 串 格 式 化 为 货币 字符 串 4 
nl langinfo() 返回 指定 的 本 地 信息 4 
nl2brO 在 9 串 中 的 每 个 新 行 之 前 插入 HIML 
换行 
number format() 通过 千 位 分 组 来 格式 化 数字 3 
ord0) 返回 字符 串 第 一 个 字符 的 ASCI 值 3 
parse_str() 把 查询 字符 串 解析 到 变量 中 3 
print() 输出 一 个 或 多 个 字符 串 k 
printf() 输出 格式 化 的 字符 串 3 
quoted printable decode0) 3 
quotemeta() 3 
rtrim() 预定 义 : 3 
setlocale() 设置 地 区 信息 (地 域 信息 ) 3 
shal() 计算 字符 串 的 SHA-1 散 列 4 
shal_file() 计算 文件 的 SHA-1 散 列 4 
similar text() 计算 两 个 字符 串 的 匹配 字符 的 数目 3 
soundex() 计算 字符 串 的 soundex 键 3 
sprintf() 把 格式 化 的 字符 串 写 入 一 个 变量 中 3 
sscanf() 根据 指定 的 格式 解析 来 自 一 个 字符 串 的 输入 4 
str_ireplace() 替换 字符 串 中 的 一 些 字符 (对 大 小 写 不 敏感 ) 名 
str_ pad() 把 字符 串 填 充 为 新 的 长 度 4 
str_repeat() 字符 串 重复 指定 的 次 数 4 
str_replace() 替换 字符 串 中 的 一 些 字符 (对 大 小 写 敏感 ) 
str_rot13() 对 字符 串 执行 ROT13 编码 4 
str_shuffle() 随机 地 打 乱 字符 串 中 的 所 有 字符 4 
str_split() 把 串 分 割 到 数组 中 5 
str_word count() 计算 字符 串 中 的 单词 数 4 
strcasecmp() 比较 两 个 字符 串 ( 对 大 小 写 不 敏感 ) 3 
po 搜索 字符 串 在 另 一 字符 串 中 的 第 一 次 出 现 


Strstr0 的 别名 
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( 续 表 ) 
支持 该 函数 的 最 早 的 
PHP 版 本 


strcmpO 


比较 两 个 字符 串 ( 对 大 小 写 敏 感 ) 


strcoll() 
strespn() 


strip_tags() 
stripcslashes() 
stripslashes() 


stripos() 


stristr() 
strlen() 


strnatcasecmp() 


strnatcmp() 


strncasecmp() 
strncmp() 
strpbrk() 


strpos() 
strrchr() 


strrev() 


strripos() 


strrpos() 
strspn() 
strstr() 
strtok() 


比较 两 个 字符 串 (根据 本 地 设置 ) 


返回 字符 串 在 另 一 字符 串 中 第 一 次 出 现 的 
位 置 (大 小 写 不 敏感 ) 

查找 字符 串 在 另 一 字符 串 中 第 一 次 出 现 的 
位 置 (大 小 写 不 敏感 

返回 字符 串 的 长 度 


使 用 一 种 “自然 ”算法 来 比较 两 个 字符 串 
(对 大 小 写 不 敏感 ) 

使 用 一 种 “自然 ”算法 来 比较 两 个 字符 串 
(对 大 小 写 敏感 ) 


前 n 个 字符 的 字符 串 比较 (对 大 小 写 不 敏感 ) 

前 n 个 字符 的 字符 串 比 较 ( 对 大 小 写 敏感 ) 

在 字符 串 中 搜索 指定 字符 中 的 任意 一 个 

返回 字符 串 在 另 一 字符 串 中 首次 出 现 的 位 
置 (对 大 小 写 敏 感 ) 

查找 字符 串 在 另 一 个 字符 串 中 最 后 一 次 出 

现 的 位 置 

反 转 字符 串 

查找 字符 串 在 另 一 字符 串 中 最 后 出 现 的 位 
置 (对 大 小 写 不 敏感 ) 

查找 字符 串 在 另 一 字符 串 中 最 后 出 现 的 位 
置 (对 大 小 写 敏感 ) 

返回 在 字符 串 中 包含 的 特定 字符 的 数目 
搜索 字符 串 在 另 一 字符 串 中 的 首次 出 现 
对 大 小 写 敏感 

把 字符 串 分 割 为 更 小 的 字符 串 


strtolower() 


把 字符 串 转换 为 小 写 


strtoupper() 


把 字符 串 转 换 为 大 写 


strtr() 


转换 字符 串 中 特定 的 字符 
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画 数 描 述 支持 该 函数 的 最 早 的 


substr() 返回 字符 串 的 一 部 分 
substr compare() 从 指定 的 开始 长 度 比较 两 个 字符 串 
substr_count() 计算 子 串 在 字符 串 中 出 现 的 次 数 


substr_replace() 把 字符 串 的 一 部 分 蔡 换 为 另 一 个 字符 串 


从 字符 串 的 两 端 删除 空白 字符 和 其 他 预定 
义 字符 


uefirst) 
vewords0 
vsprintfO 


按照 指定 长 度 对 字符 串 进行 折 行 处 理 
下 面 列举 出 最 常用 的 几 个 string 函数 。 


6.6.1 ”字符 串 加 密 : md5() 函 数 


md5() 函 数 的 定义 和 用 法 如 下 。 

。 md50 函 数 计 算 字 符 串 的 MD5 散 列 。 

。 md50 函 数 使 用 RSA 数据 安全 ， 包 括 MD5 报 文摘 译 算法 。 

如 果 成 功 ， 则 返回 所 计算 的 MD5 散 列 ， 如 果 失 败 ， 则 返回 false。 代 码 如 下 。 


md5(string,raw) 
其 中 ， 各 参数 的 说 明 如 表 6-8 所 示 。 


必需 。 规 定 要 计算 的 字符 串 

可 选 。 规 定 十 六 进 制 或 二 进 制 输出 格式 : 
。 TRUE -原始 。16 字符 二 进 制 格 式 

。 FALSE -默认 。32 字符 十 六 进 制 数 
注释 : 该 参数 是 PHP 5.0 中 添加 的 


示例 代码 如 下 。 


<?php 
$str = "Hello"; 
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echo md5($str); 
?> 


运行 结果 如 图 6-15 所 示 。 


到 http://127.0.0.1/2.19. php — Mie 
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[SET i 


图 6-15 


6.6.2 ”字符 串 重 复 操作 : str_repeat() 函 数 


str_repeat() 函 数 的 定义 和 用 法 如 下 。 
e@ ”str_repeat() 函 数 把 字符 串 重复 指定 的 次 数 。 
语法 如 下 。 
str_repeat(string,repeat) 
其 中 ， 各 参数 的 说 明 如 表 6-9 所 示 。 


必需 。 规 定 要 重复 的 字符 串 
必需 。 规 定 字符 串 将 被 重复 的 次 数 。 必 须 大 于 等 于 0 


Tepeat 
示例 代码 如 下 。 
<?php ”echo str_repeat(" 武 汉 ",3)?> 

运行 结果 如 图 6-16 所 示 。 


127.0.0.1/2_20. php 


武汉 武汉 武汉 
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6.6.3 字符 串 查找 操作 : strstr() 函 数 


strstr() 函 数 的 定义 和 用 法 如 下 。 

e strstr0 函 数 搜索 一 个 字符 串 在 另 一 个 字符 串 中 的 第 一 次 出 现 。 

该 函数 返回 字符 串 的 其 余部 分 (从 匹配 点 )。 如 果 未 找到 所 搜索 的 字符 串 ， 则 返回 
false。 

语法 如 下 。 


strstr(string,search) 
其 中 ， 各 参数 说 明 如 表 6-10 所 示 。 
表 6-10 
必需 。 规 定 被 搜索 的 字符 串 


必需 。 规 定 所 搜索 的 字符 串 。 如 果 该 参数 是 数字 ,， 则 搜索 匹配 数字 ASCII 
值 的 字符 


该 函数 是 二 进 制 安全 的 。 该 函数 对 大 小 写 敏 感 。 如 需 进 行 大 小 写 不 敏感 的 ! 


搜索 ， 请 使 用 stristr()。 


示例 代码 如 下 。 


<?php 
echo strstr("Hello 中 国 !"," 中 国 "): 
人 


运行 效果 如 图 6-17 所 示 。 


ht /N27 0 0 V2 21 oh 


图 6-17 


6.6.4 字符 串 替 换 操 作 : str_replace() 函 数 


str_ replace() 函 数 使 用 一 个 字符 串 蔡 换 字符 串 中 的 另 一 些 字符 。 
语法 代码 如 下 。 
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str_replace(find,replace,string,count) 
其 中 ， 各 参数 说 明 如 表 6-11 所 示 。 


表 6-11 


该 函数 对 大 小 写 敏 感 。 请 使 用 str_ireplace() 执 行 对 大 小 写 不 敏感 的 搜索 。 
该 函数 是 二 进 制 安全 的 。 
示例 代码 如 下 。 
<?php 


echo str_replace("world","John","Hello world!"); 
9> 


Hello John! 


在 本 例 中 ， 我 们 将 演示 带 有 数组 和 count 变量 的 str_replace() 函 数 。 


<?php 

Sarr = array("blue","red","green","yellow"); 
Print r(str replace("red","pink",$arr,$i)): 
echo "<br>"; 

echo "Replacements: $1"; 

?> 


结果 如 图 6-19 所 示 。 
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» Pip 一 时 cry=*Et Interaet EepL 
四 ED Ey | To 
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天 生男 峰 http://127.0.0 172 23 pm 
人 => .blue [1] => pink [2] => grear [3] => yellow ) 


6.6.5 字符 串 分 解 操作 : str_split() 函 数 


str_split() 函 数 的 定义 和 用 法 如 下 。 
e ”str_split() 函 数 把 字符 串 分 割 到 数组 中 。 
语法 代码 如 下 。 
str_split(string,length) 
其 中 ， 各 参数 说 明 如 表 6-12 所 示 。 


必需 。 规 定 要 分 割 的 字符 串 
可 选 。 规 定 每 个 数组 元 素 的 长 度 。 默 认 是 1 


如 果 length 值 小 于 1，str_splitO 函 数 将 返回 false。 如 果 length 值 大 于 字符 1 
串 的 长 度 ， 整 个 字符 串 将 作为 数组 的 唯一 元 素 返回 。 ! 


示例 代码 如 下 。 


<?php 
Print_r(str_split("Hello")); 
9 


输出 结果 如 图 6-20 所 示 。 
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示例 代码 如 下 。 


<?php 
Print rstr_ split("Hello",3): 
ee 


运行 效果 如 图 6-21 所 示 。 


6.6.6 字符 串 分 解 成 单词 : str_word_count() 函 数 


str_word_count() 函 数 的 定义 和 用 法 如 下 。 
estr_word_count() 函 数 计算 字符 串 中 的 单词 数 。 
语法 代码 如 下 。 

str_ word_count(string,return,char) 
其 中 ， 参 数 说 明 如 表 6-13 所 示 。 


必需 。 规 定 要 检查 的 字符 串 

可 选 。 规 定 str_word_count0 函 数 的 返回 值 

可 能 的 值 如 下 : 

。， 0- 默认 。 返 回 找到 的 单词 的 数目 

， 1- 返回 包含 字符 串 中 的 单词 的 数组 

。2- 返回 一 个 数组 ， 其 中 的 键 是 单词 在 字符 串 中 的 位 置 ， 值 是 实际 的 单词 
可 选 。 规 定 被 认定 为 单词 的 特殊 字符 。 该 参数 是 PHP 5.1 中 新 加 的 


示例 1 代码 如 下 。 


<?php 


echo str_word_count("Hello world!"): 
?> 


运行 结果 如 图 6-22 所 示 。 
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示例 2 代码 如 下 。 


<?php 

/请 看 下 面 参数 1 和 2 的 区 别 
Print_r(str_word_count("Hello world!",1)): 
echo "<br>"; 

Print r(str_word_count("Hello world!",2)):; 
?> 


结果 如 图 6-23 所 示 。 
Os- © dW um tm ODO.s 0- 
地 址 四 ) http://127.0.0.1/2.27. php 


Array ( [0] => Hello [1] => world ) 
Array ( [0] => Hello [6] => world ) 


6.6.7 ”字符 串 长 度 : strlen() 函 数 


strlen() 函 数 的 定义 和 用 法 如 下 。 
e@ ”strlen() 函 数 返 回 字符 串 的 长 度 。 
示例 代码 如 下 。 


<?php 
echo strlen("Hello world!"); // 输 出 结果 为 12 
= 


6.6.8 获取 子 字符 串 : substr() 函 数 


substrO) 函 数 的 定义 和 用 法 如 下 。 
esubstr0 函 数 返回 字符 串 的 一 部 分 。 


图 Internet 
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语法 如 下 。 
substr(string,start,length) 
其 中 ， 各 参数 说 明 如 表 6-14 所 示 。 


表 6-14 


必需 。 规 定 要 返回 其 中 一 部 分 的 字符 串 

必需 。 规 定 在 字符 串 的 何 处 开始 

” 正 数 - 在 字符 串 的 指定 位 置 开 始 

”负数 - 在 从 字符 串 结尾 的 指定 位 置 开 始 

， 0- 在 字符 串 中 的 第 一 个 字符 处 开始 

可 选 。 规 定 要 返回 的 字符 串 长 度 。 默 认 是 直到 字符 串 的 结尾 
*。 正 数 - 从 start 参数 所 在 的 位 置 返回 

” 负数 - 从 字符 串 末 端 返 回 


示例 代码 如 下 。 
<?php 


echo substr("my name is wangming",3,4); 
?> 


输出 结果 如 图 6-24 所 示 。 


【单元 小 结 】 


e@ PHP 数组 
e 字符 串 的 常用 操作 
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【单元 自 测 】 


1. 索引 数组 的 键 是 ( 。”)， 关 联 数组 的 键 是 (。”)。 
A. 浮 点 ， 字 符 串 B. 正 数 ， 负 数 
C. 字符 串 ， 布 尔 值 D. 整 型 ， 字 符 串 
2. 考虑 如 下 数组 ， 怎 样 才 能 从 数组 Smulti_array 中 找 出 值 cat? ( 


<?php 
Smulti array=array("red","green",42=>"blue","yellow"=>array("apple",9=>"pear","banana", 
"orange" 一 array("dog","cat","iguana"))); 


) 


2 


A. $multi array['yellow']['apple'][0] 
B. $multi array['blue'][0]['orange'][1] 
C. $multi array['yellow']['orange']['cat'] 
D. $multi array['yellow']['orange'][1] 
3. 运行 以 下 脚本 后 ， 数 组 $array 的 内 容 是 什么 ? ( ) 


<?php 

Sarray = array (1", 1); 
foreach ($array as $k => $v) { 
$v=2; 

} 

?> 


A. array (2', '2) B. array (1', '1) 
C. array (2, 2) D. array (Null, Null) 
. 对 数组 进行 升序 排序 并 保留 索引 关系 ， 应 该 用 哪个 函数 ? ( 
A. ksort() B. ssort() 
C. krsort() D. sortO 
. 以 下 脚本 将 按 什 么 顺序 输出 数组 Sarray 内 的 元 素 ?( ) 


<?php 

Sarray = array('al', 'a3', 'a5', 'al0', 'a20"); 
natsort($array); 

var_dump($array); 

人 > 


A. al, a3, a5, al0, a20 


B.al, a20, a3, a5, al0 
Cal0 al.a20.a3.,.a8 


D. al, al0, a5, a20, a3 
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【上 机 实战 】 


上 机 目标 


。 掌握 PHP 数组 的 使 用 
。 掌握 字符 串 操作 


上 机 练习 


9 第 一 阶段 多 


练习 : 采用 PHP 提供 的 不 同 的 数组 排序 函数 , 分 别 对 字母 、 数字 、 
数组 的 值 、 数 组 的 键 值 以 升序 排序 ， 采 用 自 定义 的 排序 函数 对 多 
维 数组 进行 升序 排序 ， 对 数组 进行 随机 处 理 。 


【问题 分 析 】 
参考 代码 如 下 。 


<?php 
// 数 组 按 字母 或 是 数字 的 升序 (从 低 到 高 ) 来 进行 排序 
$name = array("Clalei","Bill","Aala"); 
sort($name): 
for($i=0:$i<3:$i++){ 
echo $name[$i]; 


} 


echo "<br />"; 

Sprice = array(100,50.10):; 

sort(Sprice); 

for($i=0;$1<3:;$1++){ 
echo Sprice[$i]." | "; 

让 

echo "<br />"; 
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/Wasort0 函 数 以 数组 的 值 升序 为 准 

Smix = array("cobol"=>10,"Bill"=>20,"Ada"=>100); 
asort($mix); 

Print_r($mix); 

echo "<br />"; 

/ksort0 函 数 以 数组 的 关键 字 升 序 为 准 

Smix = array("cobol"=>100,"Bill"=>20,"Ada"=>10); 
ksort($mix); 

echo "<br />"; 

Print_r(Smix); 


// 对 多 维 数组 排序 
Smix = array( 
array("A",30), 
array("B",25), 
array("C",180) 
); 
// 对 其 数字 进行 升序 排列 
function compare($x,$y){ 
if($x[1] =— $y[1) 
return 0; 
elseif($x[1] < $y[1]) 
Teturn -1; 
else 
return 1; 
} 
usort($mix,"compare"); 
echo $mix[0][1]."<br />"; 
echo $mix[1][1]."<br />"; 
echo $mix[2][1]; 


echo "<br />"; 
/随机 
shuffle(Smix); 
echo "<br />"; 
print x( Smix[0] ); 
人 


运行 结果 如 图 6-25 所 示 。 
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AalaBillClalei 
10 | 50 | 100 | 
Array ( [cobol] => 10 [Bill] => 20 [Ada] => 100 ) 


Array ( [Ada] => 10 [Bill] => 20 [cobol] => 100 ) 25 
30 


| Array ( [0] => A [1] => 30 ) 


练习 : 使 用 字符 串 常 见 操作 对 字符 串 进行 替换 、 分割 、 截取 操 作 。 


【问题 描述 】 


对 字符 串 进行 替换 、 分 市、 截取 等 操作 .。 
参考 代码 如 下 。 


<?php 

// 单 个 字符 替换 

$str=" 轻 轻 的 我 走 了 [ 逗 ] 正 如 我 轻 轻 的 来 [名 ] 我 挥 一 挥 衣 袖 [去 ] 不 带 走 一 片 云彩 [ 句 ]"; 
echo " 原 字符 申 :<b>".$str."</b><br>"; 
Sstr=str_replace("[","(",$str); 
Sstr=str_replace("]",")",$str); 

echo "字符 替换 之 后 :<b>".$str."</b><br>"; 
/字符 串 蔡 换 

$str=str_replace("( 逗 )",",",$stD; 
Sstr=str_replace("( 句 )",".",$str); 

echo "字符 申 替换 之 后 :<b>".$str."</b><br>"; 


/单个 字符 替换 的 高 级 应 用 

Sstr=" 轻 轻 的 我 走 了 [去 ] 正 如 我 轻 轻 的 来 [ 句 ] 我 挥 一 挥 衣 袖 [去 ] 不 带 走 一 片 云彩 [名]"; 
echo "原始 字符 串 :<b>".$str."</b><br>"; 

Sarrl=array("[","]"); 
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Sarr2=array("(",")"); 

$str=str replace($arr1,$arr2.$str): 

echo "字符 替换 之 后 :<b>".$str."</b><br>"; 
/字符 串 蔡 换 的 高 级 应 用 
Sarr3=array("( 豆 )","( 句 )"); 
S$arr4=array(",","."); 
S$str=str_replace($arr3,$arr4,$str); 

echo "字符 串 蔡 换 之 后 :<b>".$str."</b><br>"; 


/分 割 英 文字 符 串 

$str="See what information we have on PhpCoding.cn and share your knowledge."; 
echo " 原 字 符 串 :<b>".$str."</b><br>"; 

echo "1 .以 默认 长 度 分 割 字符 串 :<br>"; 
Sarrl=str_split($str); 

echo "---\$arr1[0] 的 值 :".$arr1[0]."<br>"; 
echo "---\$arrl[1] 的 值 :".$arrl[1]."<br>"; 
echo "---\$arr1[10] 的 值 :".$arr1[10]."<br>"; 
echo "2. 以 指定 长 度 为 5 分 割 字 符 串 :<br>"; 
$arr2=str_split($str,5); 

echo "---\$arrl1[0] 的 值 :".$arr2[0]."<br>"; 


echo "---\$arr1[1] 的 值 :".$arr2[1]."<br>"; 
echo "---\$arr1[5] 的 值 :".$arr2[5]."<br>"; 


echo "原始 字符 串 :<b>".$str."</b><br>"; 

// 按 各 种 方式 进行 截取 

$strl=substr($str,S): 

echo "从 第 5 个 字符 开始 取 至 最 后 :".$str1."<br>"; 
$str2=substr($str,9,4); 

echo "从 第 9 个 字符 开始 取 4 个 字符 :".$str2."<br>"; 
$str3=substr($str,-5); 

echo " 取 倒 数 5 个 字符 :".$str3."<br>"; 

// 测 试 分 割 中 文 

$str2=" 轻 轻 的 我 走 了 ， 正 如 我 轻 轻 的 来 。"; 
echo " 原 字符 申 :<b>".$str2."</b><br>"; 

echo "1. 以 指定 长 度 为 5 分 割 字符 串 :<br>": 
Sarr3=str_split($str2,5); 

echo "---\$arr3[0] 的 值 :".$arr3[0]." <br>"; 

echo "---\$arr3[1] 的 值 :".$arr3[1]." <br>"; 

echo "2. 以 指定 长 度 为 4 分割 字符 串 :<br>"; 
S$arr4=str_split($str2.4): 

echo "---\$arr4[0] 的 值 :".$arr4[0]."<br>"; 

echo "---\$arr4[0] 的 值 :".$arr4[0]."<br>"; 
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echo "---\$arr4[0] 的 值 :".$arr4[0]."<br>"; 
J 


程序 运行 结果 如 图 6-26 所 示 。 


Se 
了 , 正如 我 轻 轻 的 来 . 我 和 带 走 一 片 云彩 
来 [名 ] 我 择 一 挥 农 袖 [ 喜 ] 不 带 走 一 片 云彩 [名 ] 


了 [去 ] 正 如 我 轻 轻 的 
ee (过 ) 不 带 走 一 片 云彩 ( 句 ) 
带 走 一 片 云 彩 . 


正如 我 轻 轻 的 来. 我 挥 一 拆 衣 袖 , 不 

原 字符 串 :Scc what i we have on PhpCoding. cn and share your knowledge. 
1 以 默认 长 度 分 划 字 符 囊 : 
一 -$arrl [0] 的 值 :S 
一 -$arrl[1] 的 值 :e 
一 -$arrl[10] 的 值 :n 
2. 以 指定 长 度 为 5 分 割 字符 囊 : 
一 -$arrl[0] 的 值 :See w 
一 -arrl[1] 的 值 :hat i 
一 -garrl [5] 的 值 :ave o | 
原 串 :See what information we have un PhpCuding. cn and share your knowledge。 也 
从 第 5 个 字符 开始 取 至 最 后 :hat information we have on PhpCoding, cn and share your knowledge. 

从 第 9 个 字符 开始 取 4 个 字符 :info 


pe 
i , 生 术 入 并 证 了 ， 正 如 我 色 轻 的 来. 
19 让 民间 


【拓展 作业 】 


1. 写 一 段 程序 ， 创 建 一 个 数组 ， 其 元 素 内 容 为 从 1 到 20 的 所 有 整数 ， 并 输出 该 
数组 。 

2. 写 一 段 代 码 ， 查 找 数组 中 是 否 存在 某 一 个 指定 的 元 素 ， 如 果 存 在 则 返回 数组 的 
索引 。 


sa 加 


PHP 面向 对 泉 


时 课程 目标 


P 掌握 PHP 的 类 、 对 象 以 及 成 员 属性 和 成 员 方法 
PP 掌握 继承 、 接 口 和 多 态 
P 掌握 PHP5 中 的 魔术 方法 
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Pun 


PHP 从 PHP3 就 开始 支持 面向 对 象 的 程序 设计 (OOP)， 但 其 对 面向 对 象 编程 的 支 
持 非常 简单 。 直 到 PHP5， 重 新 设计 了 面向 对 象 的 模型 ， 增 加 了 大 量 的 特性 ， 才 全 面 
支持 面向 对 象 编程 。 


7.1 PHP 由 的 类 和 对 象 


对 象 (Objecb 是 问题 域 或 实现 域 中 某 些 事物 的 一 个 抽象 。 它 反映 该 事物 在 系统 中 需 
要 保存 的 信息 和 发 挥 的 作用 。 它 是 一 组 属性 和 有 权 对 该 属性 进行 相关 操作 的 一 组 服务 
的 一 个 封装 体 。 关 于 对 象 要 从 两 方面 理解 : 一 方面 是 指 系统 所 要 处 理 的 现实 世界 中 的 
对 象 ， 另 一 方面 是 指 计算 机 不 直接 处 理 的 对 象 ， 而 是 处 理 相应 的 计算 机 表示 ， 这 种 计 
算 机 表示 也 称 为 对 象 ， 也 就 是 “一 切 皆 对 象 ”。 一 个 人 可 以 是 一 个 对 象 ， 一 辆 汽车 也 可 
以 是 一 个 对 象 。 当 这 些 对 象 可 以 使 用 数据 直接 进行 表示 时 ， 就 将 这 些 数据 称 为 属性 。 
一 个 人 有 姓名 ， 也 有 年 龄 ， 这 个 姓名 和 年 龄 就 是 人 的 属性 。 

面向 对 象 (OO) 是 指 把 软件 组 织 成 一 系列 离散 的 、 合 并 了 数据 结构 和 行为 的 对 象 。 
而 传统 的 软件 开发 方法 中 数据 结构 和 行为 只 是 松散 关联 的 。 现 在 面向 对 象 的 这 种 概 
念 和 方法 已 不 再 只 是 用 于 软件 开发 上 ， 也 扩展 到 很 多 方面 ， 如 数据 库 系 统 、 应 用 平 
台 、 分 布 式 管理 平台 、 人 工 智能 等 领域 。 

面向 对 象 的 程序 设计 (OOP) 旨 在 创建 软件 重用 代码 ， 以 具备 更 好 的 模拟 现实 世界 
环境 的 能 力 ， 这 使 它 被 公认 为 是 自 上 而 下 编程 的 优胜 者 。 它 通过 给 程序 中 加 入 扩展 语 
句 ， 把 函数 “封装 ” 进 编程 所 必需 的 “对 象 ”中 。 


7.1.1 声明 类 和 属性 、 方 法 的 定义 


PHP5 与 Java、C# 中 的 声明 类 一 样 ， 使 用 class 来 定义 一 个 类 ， 后 面 跟 类 名 (类 
名 可 以 是 任何 非 PHP 保留 字 的 字符 串 )， 然 后 再 跟 一 对 大 括号 ， 括 号 里 面包 含 类 的 
成 员 和 方法 定义 。 其 语法 如 下 。 


class className{ 
/成员 … 
/方法 … 


定义 类 的 方法 ， 首 先 指定 该 方法 的 访问 权限 ， 然 后 跟 关键 字 function， 接 着 是 方法 
名 称 及 参数 列表 ， 最 后 跟 大 括号 ， 大 括号 内 是 方法 体 。 语 法 如 下 。 


public function functonName( 参 数列 表 ){ 
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方法 体 
} 


下 面 的 示例 定义 一 个 Book 类 ， 并 定义 类 中 的 属性 和 方法 ， 代 码 如 下 。 


class Book { 
Private $bookId; 
private $bookName:; 


public function setBookId($bookId) { 
S$this->bookId = $bookId; // 这 里 $this->bookId 类 似 于 Java 中 的 this-bookId, this 
表示 对 对 象 本 身 的 引用 
} 
public function getBookId0 { 
return $this->bookId: 
} 


public function getBookName() { 
return $this->bookName:; 

} 

public function setBookName($bookName) { 
S$this->bookName = $bookName; 

} 

} 


在 上 面 的 示例 中 , 给 Book 类 定义 了 bookId 和 bookName 两 个 属性 ， 其中, private 
为 属性 的 访问 修饰 符 。 然 后 定义 了 4 个 公共 的 方法 ， 分 别 采用 S$this 对 对 象 Book 本 身 
的 bookId 和 bookName 进行 赋值 和 取 值 。 


7.1.2 构造 函数 和 类 的 实例 化 


构造 方法 是 一 个 名 为 _construct() 的 方法 。PHP 的 构造 方法 的 作用 与 C#、Java 中 的 
构造 方法 作用 一 样 ， 若 在 类 中 定义 该 函数 ， 它 将 被 自动 调用 。 通 常 构造 方法 用 于 自动 
执行 对 象 的 初始 化 操作 ， 如 对 象 属性 的 初始 化 。 若 在 初始 化 时 需要 使 用 外 部 数据 ， 可 
采用 给 构造 函数 添加 参数 的 方式 ， 构 造 方法 默认 为 公共 方法 ， 并 且 不 能 将 其 定义 为 私 
有 方法 或 受 保护 方法 。 

PHP 中 类 的 实例 化 与 Java、C# 中 一 样 ， 使 用 new 关键 字 。 

下 面 修 改 上 例 使 用 构造 方法 在 对 象 初始 化 时 给 属性 赋值 ， 代 码 如 下 。 


<2php 
class Book { 

Private SbookId: 
Private $bookName: 
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public function _construct($bookId,$bookName){ 
S$this->bookId=$bookId; 
S$this->bookName=$bookName; 

B 


public function getBookId0 { 
return $this->bookId; 
} 


public function getBookName() { 
Teturn $this->bookName; 
} 
} 
$bool = new Book(112,"PHP 宝典 "); 
echo "书本 编号 : ".$boo1->getBookId0."<br>"; 
echo "书本 名 称 : ".$boo1->getBookName(); 
?> 


代码 运行 结果 如 图 7-1 所 示 。 


7.1.3 ” 析 构 函数 


析 构 函数 与 构造 函数 的 功能 恰好 相反 。 析 构 函数 是 一 个 名 为 _destruct(O) 的 函数 ， 它 
是 在 对 象 被 注销 时 所 调用 的 , 通常 PHP 会 在 所 有 请 求 都 结束 时 自动 释放 该 对 象 所 占有 
的 资源 ， 所 以 析 构 函数 显得 并 不 是 特别 重要 。 但 是 在 某 些 情况 下 还 是 很 有 用 处 的 ， 如 
释放 指定 的 资源 或 者 记录 日 志 信 息 。 

在 上 例 的 基础 上 加 析 构 函数 ， 代 码 如 下 所 示 。 


<?php 

class Book { 

Private $bookId; 

private $bookName; 

/构造 函数 

public function __construct($bookId,$bookName){ 
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S$this->bookId=$bookId: 
S$this->bookName=$bookName; 
} 


public function getBookIdO) { 
return $this->bookId; 
上 


public function getBookName() { 
Teturn $this->bookName; 
} 
public function destructO{ ”// 析 构 函 数 
echo "<br> 对 象 被 释放 ! "; 
} 
} 
$bool = new Book(112,"PHP 宝典 "); 
echo "书本 编号 : ".$boo1->getBookId0."<br>"; 
echo "书本 名 称 : ".$boo1->getBookName(); 
ge 


程序 运行 结果 如 图 7-2 所 示 。 


书本 编号 ，112 
书本 名 称 ，PHP 宝 : 
对 象 被 释放 1 


7.1.4 类 的 常量 


在 PHP 中， 存在 全 局 常量 ， 使 用 define 关键 字 进 行 定义 。 而 在 PHP5 中 ， 还 可 以 
在 类 中 定义 常量 。 类 的 常量 不 属于 任何 类 的 实例 ， 只 属于 类 的 本 身 。 定 义 类 的 常量 很 
简单 ， 只 需要 使 用 const 关键 字 进 行 定 义 即 可 。 语 法 如 下 。 


const PI = 3.14; /注意 ， 类 中 的 常量 是 大 小 写 敏感 的 


类 中 的 常量 与 PHP 中 的 全 局 常量 一 样 ， 定 义 后 不 可 对 其 再 做 赋值 操作 。 

对 于 类 的 常量 的 引用 可 分 为 两 种 情况 : 一 种 是 在 类 的 内 部 进行 引用 ， 另 一 种 是 在 
类 的 外 部 进行 引用 。 不 管 是 何 时 引用 类 的 常量 ， 都 需要 使 用 范围 解析 操作 符 (::)。 在 类 
内 部 引用 类 中 的 常量 , 需要 使 用 PHP 中 的 特殊 关键 字 self 再 加 范围 解析 操作 符 再 加 类 
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中 定义 的 常量 名 ，self 表示 其 类 本 身 ， 语 法 如 下 。 
self ::PI 


如 果 是 在 类 的 外 部 引用 类 中 的 常量 ， 则 需要 使 用 类 名 加 范围 解析 符 (:)， 再 加 所 定 
义 的 常量 名 。 语 法 如 下 。 


类 名 :: PI 
在 上 例 的 基础 上 添加 常量 ， 演 示 在 类 中 和 类 外 面 引用 类 中 的 常量 ， 代 码 如 下 。 


<?php 

class Book { 

const PUBLISH = "清华 大 学 出 版 社 "; 

Private $bookId; 

Private $bookName; 

// 构 造 函 数 

public function __construct($bookId,$bookName){ 
S$this->bookId=$bookId; 
Sthis->bookName=$bookName; 

} 


public function getPublishO{ 
return self::PUBLISH; /这 里 使 用 self 引用 类 中 所 定义 的 常量 
} 
public function getBookId() { 
return $this->bookId: 
} 


public function getBookName() { 
return $this->bookName; 

} 

} 

class ITBook extends Book{ 

Private $bookPrice; 

public function getBookPrice() { 
Teturn $this->bookPrice: 

} 


function _construct($bookId, $bookName,$bookPrice){ 
parent::_ construct($bookId, $bookName): 
S$this->bookPrice = $bookPrice: 
上 
; 
enoD 
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S$bookl = new Book(112,"PHP 宝典 "); 

echo "书本 编号 : ".Sbookl->getBookId0."<br>": 

echo "书本 名 称 : ".$Sbook1->getBookName()."<br>"; 

echo "出 版 社 : "Book::PUBLISH."<br>"; 。 // 这 里 在 类 外 部 通过 类 名 :: 引 用 常量 
echo "------------------ 子 类 : IT 类 图 书 。-------------------<br>"; 
$itBookl = new ITBook("102", "SQLSERVER 高 级 ", "了 45.5 元 "); 
echo "书本 编号 : ".$itBook1->getBookId0."<br>"; 

echo "书本 名 称 : ".$itBook1->getBookName()."<br>"; 

echo "书本 价格 : ".$itBook1->getBookPrice()."<br>"; 

echo "出 版 社 : ".$itBook1->getPublish()."<br>"; 

?> 


程序 代码 运行 结果 如 图 7-3 所 示 。 


es 父 类 : 图 书信 息 
书本 痊 号 ，112 


书本 名 称 ，PHP 宝 典 
出 版 社 ， 清 华 大 学 出 版 社 
二 于 类: TI 类 图 书 
书本 编号 ，102 


SQL: 
书本 价格 ， 关 45.5 元 
出 版 社 ， 清华 大 学 出 版 社 


7.2 访问 类 由 的 方法 和 属性 


7.2.1 访问 修饰 符 


PHP 中 访问 保护 通过 3 个 关键 字 来 实现 : public、protected 和 private。 在 开发 时 ， 
需要 指定 属性 或 方法 的 访问 权限 。 为 了 向 下 兼容 ， 定 义 方法 时 未 指定 访问 权限 的 将 自 
动 设 置 为 public。 

表 7-1 列 出 了 3 种 访问 修饰 符 及 说 明 。 


表 7-1 
访问 修饰 符 说 明 
公共 的 成 员 ， 可 被 所 属 类 的 成 员 以 及 不 属于 类 的 成 员 访问 
public 在 对 象 外 部 使 用 时 采用 对 象 名 -> 属性 名 或 对 象 名 -> 方法 名 访问 
在 对 象 内 部 使 用 时 采用 Sthis-> 属 性 名 或 Sthis-> 方 法 名 访问 
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访问 修饰 符 
受 保护 的 成 员 ， 可 被 所 属 类 或 子 类 访问 
ee 采用 特殊 变量 Sthis-> 受 保护 的 变量 名 或 gthis-> 受 保护 的 方法 名 进行 访问 


Private 私有 成 员 ， 仅 所 属 类 的 成 员 才 可 以 访问 


7.2.2 静态 属性 和 静态 方法 


PHP 中 可 以 定义 静态 属性 和 静态 方法 ， 类 中 的 静态 成 员 不 属于 任何 类 的 实例 ， 只 
属于 类 本 身 。 静 态 属 性 和 静态 方法 都 通过 static 关键 字 定义 。 

访问 类 的 静态 成 员 有 两 种 情况 : 一 种 是 在 类 的 内 部 访问 , 可 采用 “self:: 静 态 成 员 名 ” 
的 方式 访问 ， 另 一 种 是 在 类 的 外 部 访问 ， 可 采用 “类 名 :: 静 态 成 员 名 ”的 方式 访问 。 

下 面 的 示例 演示 了 静态 属性 和 静态 方法 的 使 用 ， 代 码 如 下 所 示 。 


<?php 
class Book{ 
static $count = 0 ; 
public function _ constructO{ 
self::addCount(); 
} 
public static function addCountO{ 
self::$count+t:; 
} 


public static function printCountO{ 
echo "数量 : " .self::$count; 

bp 

} 
echo Book::printCount|."<br>"; // 输 出 结果 0 

S$bl = new Book(); /实例 化 类 ， 这 里 将 自动 调用 构造 方法 ， 在 构造 方法 中 

调用 静态 方法 

addCount() 

echo Book::printCount(); /1/ 输 出 结果 为 1 

?> 


运行 结果 如 图 7-4 所 示 。 
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7.2.3 ”魔术 方法 


在 PHP 中 ， 除 提供 了 静态 方法 外 ， 还 提供 了 几 个 有 用 的 魔术 方法 。 这 些 魔术 方法 
包括 前 面 讲 过 的 构造 函数 construct)、 析 构 函 数 _ destruct0、 字 符 串 转换 函数 
_toString()、 克 隆 函数 _clone() 等 。 下 面 将 对 字符 串 转换 函数 和 克隆 函数 进行 讲解 。 


1. 字符 串 转换 函数 _ toString() 


如 果 想 在 采用 echo 或 print 命令 显示 对 象 时 ， 显 示 对 象 的 信息 ， 则 可 以 使 用 
_toString() 方 法 。 该 方法 用 于 返回 表示 对 象 信息 的 字符 串 ， 并 且 在 类 中 定义 了 该 方法 ， 
当 用 户 试图 输出 对 象 时 会 自动 调用 _toString() 方 法 。 

下 面 示例 演示 了 _toString() 方 法 的 使 用 ， 代 码 如 下 。 


<?php 

class Book{ 

Private $bookId: 
Private $bookName; 


public function ”construct($bookId,SbookName){ 
$this->bookId=$bookId; 
$this->bookName=$bookName: 
} 
// 这 里 定义 _toString0 方 法 
public function _ toStringO{ 
return "编号 :".$this->bookId."<br> 名 称 : ".$this->bookName:; 
b 
} 
S$bl = new Book(101, "JAVA 语言 基础 "); 
print $bl; ”// 在 这 里 输出 对 象 时 ， 会 自动 调用 _toString0 方 法 
> 


程序 运行 结果 如 图 7-5 所 示 。 


2. 克隆 函数 _clone() 
在 实际 应 用 中 ， 除 了 采用 new 关键 字 创 建 对 象 外 ， 还 可 以 使 用 clone 关键 字 实现 
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对 象 的 克隆 ， 所 克隆 的 对 象 将 拥有 原 对 象 的 所 有 属性 ， 语 法 格式 如 下 。 
ScloneObject = colone $object: 


其 中 ，$object 为 被 克隆 的 对 象 ，$cloneObject 为 克隆 出 的 对 象 ， 它 将 拥有 被 克隆 对 象 
$object 的 全 部 属性 。 


修改 上 例 ， 使 用 对 象 $bl 克隆 出 一 个 对 象 Sb2， 示 例 代码 如 下 。 
<2php 
class Book{ 
Private $bookId; 
Private $bookName; 


public function __construct($bookId,$bookName){ 
S$this->bookId=$bookId; 
S$this->bookName=$bookName;: 


} 


A rn rai 
return "编号 :".$this->bookId."<br> 名 称 : ".$this->bookName: 


S$bl = new Book(101, "JAVA 语言 基础 "); 


克隆 对 象 $b2 ------- -<br>'; 
$b2 = clone $bl; /克隆 对 象 
Print $b2; 
?> 


程序 运行 结果 如 图 7-6 所 示 。 


PHP 还 提供 了 一 种 clone0O 魔 术 方法 ， 可 以 在 克隆 对 象 时 自动 调用 该 方法 。 
下 面 示例 用 到 了 clone0 方 法 ， 代 码 如 下 。 


<?php 
class Book{ 
Private $bookId; 
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Private $bookName: 


public function _construct($bookId,$bookName){ 
S$this->bookId=$bookId; 
Sthis->bookName=$bookName; 

b 


public function _ toStringO{ 
return "编号 :".$this->bookId."<br> 名 称 : ".$this->bookName; 
} 
public function cloneO{ 
Sthis->bookId++; 
} 
} 
echo ~--------- ”对象 $bl ----------<br>' 
$bl = new Book(101, "JAVA 语言 基础 "); 
print $bl; 
echo '<br>---------- ”克隆 对 象 $b2 ----------<br>'; 
$b2 = clone $bl; ”// 克 隆 对 象 
print $b2; // 由 于 对 象 在 克隆 时 调用 了 _clone0 方 法 ， 所 有 bookId 的 值 将 加 1 
i 


运行 结果 如 图 7-7 所 示 。 


和 http://127 0.0.1/Deno2/test. pop 


7.3 类 的 继 稀 


继承 是 指 一 个 对 象 直接 使 用 另 一 个 对 象 的 属性 和 方法 。 如 果 一 个 类 继承 自 另 一 个 
类 ， 则 将 这 个 类 称 为 子 类 或 者 扩展 类 ， 将 被 继承 的 类 称 为 父 类 或 基 类 。 使 用 继承 ， 使 
子 类 具有 其 父 类 的 所 有 可 访问 的 方法 和 属性 ， 而 不 用 再 次 编写 相同 的 代码 。 

子 类 在 继承 父 类 时 ， 可 以 重新 定义 某 些 方法 和 属性 ， 即 在 子 类 中 覆盖 父 类 的 原 有 
属性 和 方法 ， 使 其 与 父 类 具有 相同 方法 名 而 具有 不 同 的 功能 。 在 实际 的 应 用 中 ， 使 用 
继承 的 主要 目的 在 于 实现 代码 的 重用 。 
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7.3.1 继承 方法 


PHP 的 继承 和 Java 一 样 ， 使 用 关键 字 extends， 被 继承 的 类 称 为 基 类 或 父 类 ， 继 
承 的 类 称 为 扩展 类 或 子 类 。 通常 在 PHP 中 使 用 parent 表示 父 类 ， 常 用 于 访问 父 类 的 方 
法 和 属性 。 使 用 self 表示 当前 类 ， 常 用 于 访问 当前 类 中 的 常量 、 静 态 变量 和 方法 。 

在 使 用 parent 和 self 访问 方法 和 属性 时 都 需要 在 其 后 加 范围 解析 操作 符 。 范 围 解 
析 操 作 符 (::) 也 称 为 Paamayim Nekudotayim， 也 可 以 说 是 双 冒 号 。 这 是 Zend 开发 小 组 
在 开发 Zend 引擎 时 所 做 出 的 决定 。 实 际 上 Paamayim Nekudotayim 在 希 伯 来 文 就 是 双 
冒号 的 意思 。 

下 面 演示 类 的 继承 使 用 方法 ， 代 码 如 下 。 


<?php 

class Book { 

Private SbookId; 

Private $bookName': 

/构造 函数 

public function construct($bookId,SbookName){ 
$this->bookId=$bookId: 
S$this->bookName=$bookName;: 

} 


public function getBookId() { 
return $this->bookId; 
} 


public function getBookName() { 
return $this->bookName; 
} 


} 
class ITBook extends Book{ 
Private $bookPrice; 
public function getBookPrice() { 
Teturn $this->bookPrice; 
} 


function _construct($bookid, $bookname,$bookprice){ 
// 这 里 parent:: 类 似 于 Java 中 的 super 关键 字 ， 或 者 C# 中 的 base 关键 字 
parent::_ construct($bookId, SbookName): 
Sthis->bookPrice = $bookPrice; 
} 


} 
elo Ia 
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S$bookl =new Book(112,"PHP 宝典 "); 

echo "书本 编号 : ".$book1->getBookId(."<br>"; 

echo "书本 名 称 : ".$Sbook1->getBookName()."<br>"; 

echo "------------------ 了 类: IT 类 图 书 。-------------------<br>"; 
S$itBookl = new ITBook("102", "SQLSERVER 高 级 ", "45.5 元 "); 
echo "书本 编号 : ".$itBook1->getBookId0."<br>"; 

echo "书本 名 称 : ".$itBook1->getBookName()."<br>"; 

echo "书本 价格 : ".SitBookl->getBookPrice0."<br>"; 

po 


运行 后 结果 如 图 7-8 所 示 。 


7.3.2 ”通过 魔术 方法 实现 “ 重 载 ” 


PHP 所 提供 的 “ 重 载 ”(overloading) 是 指 动态 地 “创建 ”类 属性 和 方法 ， 是 通过 
魔术 方法 (magic methods) 来 实现 的 。 

当 调 用 当前 环境 下 未 定义 或 不 可 见 的 类 属性 或 方法 时 ， 重 载 方法 会 被 调用 。 本 节 
后 面 将 使 用 “不 可 访问 成 员 (inaccessible members)” 和 “不 可 访问 方法 (inaccessible 
methods)” 来 称呼 这 些 未 定义 或 不 可 见 的 类 属性 或 方法 。 

所 有 的 重 载 方法 都 必须 被 声明 为 public。 


PHP 中 的 “ 重 栽 ” 与 其 他 绝 大 多 数 面向 对 象 语言 不 同 。 传 统 的 “ 重 栽 ”用 j 
于 提供 多 个 同名 的 类 方法 ， 但 各 方法 的 参数 类 型 和 个 数 不 同 。 


1. 属性 重 载 ( 见 表 7-2) 


Void _set (string $name, mixed 


$value) 


mixed get (string Sname) 读 取 未 定义 的 变量 的 值 时 ，_getO 会 被 调用 


在 给 未 定义 的 变量 赋值 时 ，_setO 会 被 调用 
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bool isset (string $name) 当 对 未 定义 的 变量 调用 isset0 或 empty0 时 ，_issetO 会 被 调用 
void _unset (string $name) 当 对 未 定义 的 变量 调用 unset0 时 ，_unsetO 会 被 调用 


参数 $name 是 指 要 操作 的 变量 名 称 。 set() 方 法 的 Svalue 参数 指定 了 $name 变量 
的 值 。 

属性 重 载 只 能 在 对 象 中 进行 。 在 静态 方法 中 ， 这 些 魔术 方法 将 不 会 被 调用 。 所 以 
这 些 方法 都 不 能 被 声明 为 static 。 

下 面 的 代码 演示 了 属性 的 重 载 。 


<?php 
class MemberTest { 
/*#* ”被 重 载 的 数据 保存 在 此 */ 
Private $data = array(); 
/** ” 重 载 不 能 被 用 在 已 经 定义 的 属性 */ 
public $declared = 1: 


/*#* 只 有 从 类 外 部 访问 这 个 属性 时 ， 重 载 才 会 发 生 */ 
Private $hidden = 2; 


public function _set($name, $value) { 
echo "设置 : '$name' 的 值 是 :'$value\n"; 
Sthis->data[$name] = $value; 

} 


public function get($name) { 
echo "获得 '$name' 的 值 <br>"; 
f(array_key_exists($name, $this->data)) { //array_key_exists() 函数 判断 某 个 数组 
中 是 否 存在 指定 的 key 
return $this->data[$name]: 


} 


/#*# PHP 5.1.0 之 后 版 本 */ 

public function _isset($Sname) { ”//isset0 用 于 检测 变量 是 否 设置 
echo "变量 '$name' 是 否 已 经 设置 ?m": 
return isset($this->data[$name]); 

} 


/**# ”PHP 5.1.0 之 后 版 本 */ 
public function _unset(Sname) { ”//unset 一 释放 给 定 的 变量 
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echo "释放 '$namen": 
unset($this->data[ $name]); 
} 


/** 非 魔 术 方 法 到 
public function getHiddenO { 
Teturn $this->hidden; 
} 
} 
echo "<pre><br>"; 
S$obj = new MemberTest; 


Sobj->a = 1; // 给 一 个 不 存在 的 属性 变量 赋值 时 会 调用 _set() 
echo $obj->a . "<br>"; // 读 取 未 定义 的 变量 值 时 ， 将 调用 _getO 


var_dump(isset($obj->a)); /检测 变量 是 否 已 设置 
unset($obj->a); // 释 放 变 量 
var_dump(isset($obj->a)); /再 次 检测 变量 是 否 已 设置 
echo "<br>"; 


echo $obj->declared . "\n\n"; 


echo $obj->getHidden() . "\n"; 

echo "下 面 调 用 私有 的 成 员 hidden， 程 序 会 认为 是 在 读 取 未 定义 的 变量 ， 它 将 调用 
getO<br>"; 

echo $obj->hidden . "<br>": 

echo "</pre>"; 

?> 


程序 运行 结果 如 图 7-9 所 示 。 


http://127.0.0. 1/Demo2/test.php 
设置 : “a” 的 值 是 :1" 
获得 “a 的 什 
1 
变量 "“a” 是 否 已 经 设置 ? 
bool (true) 


释放 'a 

变量 “a” 是 否 已 经 设置 ? 
bool (false) 

1 


2 
下 面 调用 私有 的 成 员 hidden， 程 序 会 认为 是 在 读 取 未 定义 的 变 重 ， 它 将 调用 _get () 
获得 “hidden 的 值 
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2. 方法 重 载 ( 见 表 7-3) 


表 7-3 
万" 法 说 明 
mixed _call (string $name ， 当 调 用 一 个 不 可 访问 方法 (如 未 定义 , 或 者 不 可 见 ) 时 ，_call0 
array $arguments) 会 被 调用 
mixed _callStatic (string $name , | ” 当 在 静态 方法 中 调用 一 个 不 可 访问 方法 (如 未 定义 ， 或 者 不 
array $arguments) 可 见 ) 时 ，_callStatic0 会 被 调用 


$name 参数 是 要 调用 的 方法 名 称 。$arguments 参数 是 一 个 数组 ， 包 含 着 要 传递 给 
方法 的 参数 。 


7.3.3 使 用 final 对 继承 和 重 载 进行 限制 


在 实际 应 用 中 ， 虽 然 可 以 以 某 个 类 进行 继承 来 实现 另 一 个 扩展 类 ， 在 扩展 类 中 可 
以 通过 重 写 父 类 中 的 方法 的 方式 实现 新 的 功能 ， 但 是 在 某 些 时候 却 需要 确保 某 一 个 方 
法 不 能 够 被 扩展 类 改写 。 此 时 可 采用 关键 字 final 访问 控制 符 声 明 一 个 方法 不 能 够 被 其 
扩展 类 进行 改写 。 

下 面 的 示例 演示 了 一 个 扩展 类 改写 其 父 类 的 final(0 方 法 ， 代 码 如 下 所 示 。 


<?php 
class father{ 
final function showO{ 
echo "这 是 : "._METHOD_; 
class son extends father { 
function showO{ 
echo "这 是 : "._ CLASS ; 
} 
} 
Sbc = new son(); 
Sbe->show(); 
?> 


在 上 面 的 示例 中 ,首先 定义 一 个 父 类 father， 然 后 定义 一 个 扩展 类 son, 在 扩展 
类 中 改写 了 父 类 中 具有 final 访问 控制 符 的 方法 display()。 由 于 该 方法 被 声明 为 
final， 因 此 不 能 在 扩展 类 中 改写 该 方法 。 程 序 在 运行 时 PHP 将 给 出 如 图 7-10 所 示 
的 错误 提示 。 
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了 atal error: Cannot override final method father::show() in 
C:\wamp\www\Test\test. php on line 11 


多 态 (polymorphism) 按 字面 意思 理解 就 是 “多 种 形态 ” 可 以 理解 为 多 种 表现 形式 ， 
即 “ 一 个 对 外 接口 , 多 个 内 部 实现 方法 ” 在 面向 对 象 的 理论 中 , 多 态 性 的 一 般 定义 为 : 
同一 个 操作 作用 于 不 同 的 类 的 实例 ， 将 产生 不 同 的 执行 结果 。 即 不 同类 的 对 象 收 到 相 
同 的 消息 时 ， 将 得 到 不 同 的 结果 。 

在 实际 的 应 用 开发 中 , 通常 为 了 使 项 目 能 够 在 以 后 的 时 间 里 轻松 实现 扩展 与 升级 ， 
需要 通过 继承 ， 实 现 可 复 用 模块 ， 从 而 轻松 升级 。 在 进行 可 复 用 模块 设计 时 ， 就 需要 
尽 可 能 地 减少 使 用 流程 控制 语句 ， 此 时 就 可 以 采用 多 态 实 现 该 类 设计 。 


下 面 的 示例 是 使 用 流程 控制 语句 实现 不 同类 的 处 理 ， 代 码 如 下 。 


<?php 
class student{ ”// 定 义学 生 类 
public function workO{ 
echo "学 生 的 工作 是 学 习 ! <br>"; 
} 
} 
class teacher{ ”// 定 义 教师 类 
public function workO{ 
echo "老师 的 工作 是 教学 ! <br>"; 
D 
} 
function printworking($obj){ /定义 处 理 函 数 
if($obj instanceof student){ // 如 果 对 象 是 student 类 ， 则 显示 学 生 工作 方法 
$obj->work(); 
} 
else if($obj instanceof teacher){ /如 果 对 象 是 teacher 类 ， 则 显示 教师 工作 方法 
$obj->work(); 
} 
else{ 
echo "对 象 错误 !<br>"; 。“// 如 果 都 不 是 以 上 类 ， 则 显示 出 错 信 息 
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Dy 
printworking(new student0);// 显 示 工 作 状 态 


i 


程序 运行 结果 如 图 7-11 所 示 。 


EIEIO 


学 生 的 工作 是 学 习 1 
老师 的 工作 是 教学 ! 


上 面 的 示例 中 ， 首 先 定义 两 个 类 student 和 teacher， 然 后 定义 一 个 处 理 函 数 
printworking()。 在 该 函数 中 判断 是 否 是 已 经 定义 的 对 象 ， 并 调用 对 象 的 work0 方 法 。 

上 面 的 程序 中 ， 如 果 要 显示 出 几 种 不 同 岗位 的 工作 状态 ， 需 要 首先 定义 类 ， 并 在 
类 中 定义 工作 work() 方 法 ， 然 后 在 printworking() 函 数 里 增加 else 站 语句 来 检查 对 象 是 
哪 一 个 岗位 类 。 这 在 实际 应 用 中 ， 是 非常 不 可 取 的 。 


如 果 采 用 多 态 ， 则 可 以 轻松 解决 此 问题 。 多 态 使 用 继承 来 解决 这 个 问题 。 子 类 继 
承 父 类 ， 并 继承 父 类 的 所 有 方法 和 属性 。 
下 面 的 示例 演示 了 多 态 的 使 用 ， 代 码 如 下 。 


<?php 
class person{ /定义 父 类 
protected function workO{f ”// 定 义 work 方法 ， 需 要 在 子 类 实现 
echo "本 方法 需要 在 子 类 中 重 写 <br>"; 
} 
} 
class student extends person { 
public function workO{ 
echo "学 生 的 工作 是 学 习 ! <br>"; 
} 
} 
class teacher extends person { 
public function workO{ 
echo "老师 的 工作 是 教学 ! <br>"; 
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if($obj instanceof person){ /如 果 是 person 对 象 ， 则 显示 其 工作 状态 
$obj->work(); 
时 
else{ 
echo "对 象 错误 !<br>"; 
} 
上 
printworking(new student()); 
printworking(new teacher()); 
?> 
程序 运行 效果 同上 例 一 样 。 


7.5 接口 


类 的 继承 可 以 描述 几 个 类 之 间 的 父子 关系 ， 如 果 需 要 使 某 一 个 类 同时 继承 自 多 个 
类 ， 由 于 PHP 不 支持 多 重 继承 ， 而 采用 了 接口 。 

在 面向 对 象 编程 中 ， 接 口 是 用 来 定义 程序 的 一 种 协议 ， 是 一 系列 方法 的 声明 ， 是 
一 些 方法 特征 的 集合 。 即 一 个 接口 只 有 方法 的 特征 而 没有 方法 的 实现 ， 所 有 实现 接口 
的 类 或 结构 都 必须 与 接口 中 的 定义 完全 一 致 。 但 实现 接口 类 中 的 这 些 方法 可 以 在 不 同 
的 地 方 被 不 同 的 类 实现 ， 而 这 些 实 现 可 以 具有 不 同 的 行为 。 

在 一 个 系统 中 ， 接 口 描述 的 是 该 系统 所 能 够 提供 的 所 有 服务 ， 但 不 包含 实现 这 些 
服务 的 具体 细节 。 类 或 结构 可 以 像 类 继承 基 类 或 结构 一 样 从 接口 继承 。 在 PHP 中 , 不 
能 从 多 个 类 进行 继承 ， 但 可 以 继承 自 多 个 接口 。 当 类 或 结构 继承 接口 时 ， 将 继承 接口 
的 成 员 定义 ， 却 不 继承 成 员 的 实现 。 若 要 实现 该 接口 ， 类 中 的 所 有 对 应 成 员 都 必须 是 
公共 的 、 非 静态 的 ， 并 且 必 须 与 接口 具有 相同 的 成 员 名称 。 


7.5.1 接口 的 实现 


在 实际 应 用 中 ， 必 须 首先 声明 接口 ， 然 后 实现 该 接口 。 在 PHP 中 ， 采 用 关键 字 
interface 声明 接口 。 声 明 接口 的 语法 格式 如 下 。 


interface interfacename{ 
public function functionname(); 

} 
其 中 ， 关 键 字 interface 是 标识 接口 的 关键 字 ，interfacename 是 声明 的 接口 名 称 ， 大 括 
号 内 为 接口 体 ， 在 该 接口 体 中 ， 包 含 方法 的 声明 ， 而 没有 方法 的 具体 实现 。 

一 个 类 采用 关键 字 implements 来 实现 某 个 接口 ， 实 现 某 个 接口 的 类 将 自动 获得 该 

接口 所 定义 的 常量 ， 并 且 必 须 为 接口 中 的 函数 原型 提供 函数 定义 。 实 现 接口 的 语法 格 
式 如 下 。 
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class className Implements interfacel , interface2 ,......{ 
} 


其 中 ， 关 键 字 implements 声明 该 类 将 实现 接口 interfacel 等 。 如 果菜 个 类 同时 实现 多 
个 接口 ， 只 需要 在 关键 字 implements 后 面 以 逗号 (,) 隔 开 即 可 。 在 实现 接口 的 类 中 ， 必 
须 定义 接口 所 声明 的 所 有 函数 。 

下 面 采 用 接口 的 方式 续 写 上 例 ， 代 码 如 下 。 


<?php 
interface IPerson{ ”// 定 义 接口 
function work():; 
} 
class student implements IPerson { 
public function work0{ ”// 实 现 接口 中 的 方法 
echo "学 生 的 工作 是 学 习 ! <br>"; 


} 
} 
class teacher implements IPerson { 
public function workO{ // 实 现 接口 中 的 方法 
echo "老师 的 工作 是 教学 ! <br>"; 
} 
b 
function printworking($obj){ 
if($obj instanceof IPerson ){ /判断 是 否 实现 了 IPerson 接口 
S$obj->work(); 
} 
else{ 
echo "对 象 错误 !<br>"; 
} 
} 
printworking(new studentO): 
printworking(new teacher()); 
?> 


7.5.2 接口 的 继承 


接口 可 以 像 类 一 样 从 其 他 接口 进行 继承 。 但 与 类 继承 不 同 的 是 ， 类 继承 只 允许 继 
承 自 一 个 父 类 , 而 接口 继承 可 以 实现 多 重 继承 。 接 口 继承 的 语法 与 类 继承 的 语法 类 似 ， 
语法 格式 如 下 。 


interface interfaceName extneds interfacel,interface2， 
/接口 体 … 
} 
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在 接口 继承 中 ， 与 类 实现 接口 的 规则 类 似 ， 接 口 只 能 够 继承 与 本 接口 不 相 冲 突 的 
接口 。 即 如 果 继 承 的 接口 定义 了 一 个 被 继承 的 接口 中 存在 的 某 个 方法 或 常量 ， 在 执行 
时 PHP 将 给 出 相应 的 错误 信息 。 


【单元 小 结 】 


。 PHP 的 类 、 对 象 以 及 成 员 属 性 和 成 员 方法 
e 继承 、 接 口 和 多 态 
e PHP5 中 的 魔术 方法 


【单元 自 测 】 


1. 如 何 即时 加 载 一 个 类 ? ( ) 
A. 使 用 _autoload() 魔 术 函 数 B. 把 它们 定义 为 forward 类 
C. 实现 一 个 特殊 的 错误 处 理 手段 D. 不 可 能 


2 提供 了 一 个 高 性 能 的 解决 面向 对 象 中 重复 出 现 的 问题 的 方案 ? 
3. 以 下 脚本 输出 什么 ?(  ) 


<?php 


{ 
function a0 
echo 'Parent called'; 


Sc = new b(); 
?> 
A. Parent called B. 一 个 错误 
人 三 个 警告 D. 什么 都 没有 
4. PHP 中 有 静态 类 变量 吗 ?( ) 
A. 有 B. 没有 
5. 以 下 脚本 输出 什么 ?(  ) 
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A.1 
十. 去 
C. 一 个 错误 ， 因 为 没有 定义 a::$myvar 
D. 一 个 警告 ， 因 为 没有 定义 a::$myvar 


【上 机 实战 】 


上 机 目标 


。 掌握 PHP 的 类 、 对 象 以 及 成 员 属性 和 成 员 方法 
。 掌握 继承 、 接 口 和 多 态 的 用 法 
。 掌握 PHP5 中 的 魔术 方法 
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上 机 练习 


9 第 一 阶段 多 


练习 : 练习 PHP 中 访问 修饰 符 以 及 继承 的 使 用 。 


【参考 步骤 】 


(1) 定义 一 个 父 类 myClass， 分 别 定 义 3 个 公共 的 属性 和 方法 、3 个 受 保 护 的 属性 
和 方法 、3 个 私有 属性 和 方法 。 

(2) 定义 一 个 从 myClass 继承 的 扩展 类 sonClass, 再 分 别 实例 化 父 类 myClass 和 子 
类 sonClass， 最 后 调用 其 公共 方法 setName() 设 置 name 属性 。 

(3) 在 父 类 myClass 中 定义 了 受 保护 方法 setSize(), 在 子 类 sonClass 中 可 以 访问 该 
方法 ， 并 且 也 可 以 改写 该 方法 。 在 子 类 sonClass 中 重 写 了 setSize() 方 法 。 

(4) 在 父 类 myClass 和 子 类 sonClass 中 都 定义 了 私有 属性 id 和 私有 方法 setId()， 
但 是 因 两 个 属性 和 方法 都 为 私有 ， 所 以 在 子 类 sonClass 中 无 法 看 到 父 类 的 私有 属性 id 
和 私有 方法 setId(), 因此 , 父 类 和 子 类 的 私有 属性 id 和 私有 方法 setId() 是 各 不 相关 的 。 

代码 如 下 。 


<2php 
class myClass{ 
public $name; 
protected $size: 
Private $id; 
public function __construct($size){ 
Sthis->setSize = $size: 
S$this->setId(); 
} 
public function setName($name){ 
S$this->name = $name; 
} 
protected function setSize($size){ 
Sthis->size = $size; 
} 
private function setIdO{ 
S$this->id = rand(10, 5000); 
了 
} 
class sonClass extends myClass{ 
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Private $id; 

public function _construct($size){ 
Sthis->setSize($size); 
Sthis->setId(); 

} 

protected function setSize($size){ 
Sthis->size = $size * 2; 

} 

Private function setIdO{ 
S$this->id = rand(1, 10); 

} 

} 

SmyClass = new myClass(15); 

SmyClass->setName("parent class"); 


$sonClass = new sonClass(15); 
$sonClass->setName("son class"): 


Print_r($myClass); 
echo "<br>"; 
print_r($sonClass); 
?> 


程序 运行 结果 如 图 7-12 所 示 。 


mClass Object ( [name] =》parent class [size:protected] => [id:private] => 
4414 [setSize] => 15 ) 

sonClass 0bject ( [id:private] =》9 [name] =》son class [size:protected] => 
30 [id:private] => ) 


图 7-12 


人 第 二 阶段 多 


练习 : 静态 属性 和 静态 方法 


【问题 描述 】 
(1) 在 PHP 中 ， 静态 属性 和 静态 方法 不 属于 某 一 个 实例 ， 只 属于 类 本 身 。 在 类 内 
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部 可 以 通过 “self 静态 属性 名 ”和 “self 静态 方法 名 ”的 形式 进行 调用 ， 在 类 外 部 可 以 
通过 “类 名 称 :静态 属性 名 ”和 “类 名 称 :静态 方法 名 ”的 形式 进行 调用 。 

(2) 下 面 的 程序 中 , 定义 了 一 个 类 myClass， 在 该 类 中 定义 一 个 静态 变量 和 两 个 静 
态 方法 ; 然后 直接 在 类 的 外 部 调用 类 里 面 的 静态 方法 printId(); 再 使 用 new 关键 字 实 
例 化 类 ， 此 时 将 自动 调用 其 构造 函数 ， 在 该 构造 函数 中 调用 了 类 的 静态 方法 addId(); 
静态 变量 将 自动 加 1， 最 后 再 调用 类 的 printId() 方 法 ， 静 态 变量 $id 加 为 1 。 


<?php 
class myClass{ 
public static $id = 0; 
public function _ constructO{ 
self::addId(); 
ly 
public static function printIdO{ 
echo "myClass id: " . self:$id: 
上 
public static function addIdO{ 
self::$id++; 
} 
b 
echo myClass::printId() . "<br>"; 
SmyClass = new myClass; 
echo myClass::printId() . "<br>"; 
?> 


程序 运行 结果 如 图 7-13 所 示 。 


【拓展 作业 】 


1. 写 一 段 代码 ， 定 义 一 个 汽车 类 ， 有 品牌 与 价格 两 种 属性 。 并 为 类 实例 化 对 象 ， 
为 对 象 的 属性 赋值 并 引用 。 

2. 在 上 例 的 基础 上 为 汽车 类 定义 一 个 跑车 子 类 。 为 子 类 实例 化 对 象 并 访问 父 类 的 
属性 。 
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时 课程 目标 


P 掌握 PHP 中 文件 的 上 传 和 下 载 方法 
P 掌握 PHP 中 的 异常 类 型 及 处 理 方法 
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Pun 


在 Web 应 用 系统 开发 中 ， 文 件 上 传 和 下 载 是 非常 常用 的 功能 ， 本 单元 介绍 PHP 
中 的 文件 上 传 和 下 载 功 能 的 实现 。 对 于 文件 上 传 ， 浏 览 器 在 上 传 的 过 程 中 是 将 文件 
以 流 的 形式 提交 到 服务 器 端 ; 对 于 文件 下 载 ， 浏 览 器 在 下 载 的 过 程 中 是 将 文件 以 流 
的 形式 从 服务 器 端 下 载 到 客户 端 。 


8.1 文件 的 上 传 与 下 载 


在 实际 应 用 中 ， 有 时 需要 用 户 从 本 地 上 传 文件 至 服务 器 进行 处 理 ， 有 时 需要 服务 
器 提供 文件 供用 户 下 载 。 下 面 对 文 件 的 上 传 和 下 载 进行 详细 讲解 。 


8.1.1 开启 上 传 功能 


Web 服务 器 具有 文件 上 传 功能 ， 必 须 在 配置 文件 php.ini 中 设置 允许 文件 上 传 。 配 
置 文件 php.ini 对 于 文件 上 传 的 选项 如 表 8-1 所 示 。 


file_ uploads 是 否 开启 文件 上 传 
upload tmp_dir 上 传 文件 临时 目录 
upload max filesize 允许 最 大 上 传 大 小 


e file uploads 选项 : 标识 是 否 允 许 文 件 上 传 。 其 默认 值 为 1， 允许 文件 上 传 。 如 果 
不 允许 服务 器 上 传 文件 ， 可 将 其 改 为 0， 再 重启 服务 器 即 可 。 如 果 不 允 许 服 务 器 
上 传 文件 ， 则 后 两 个 选项 自动 失效 。 

e ”upload_tmp_dir 选项 : 标识 上 传 的 文件 的 临时 存放 目录 。 其 默认 值 为 null， 此 
时 上 传 的 临时 文件 将 保存 至 系统 临时 目录 ， 也 可 设置 为 指定 的 目录 。 

e upload max filesize 选项 : 标识 客户 端 一 次 可 上 传 的 最 大 文件 夹 尺 寸 ， 默 认为 
2MB。 如 果 服 务 器 需要 ， 可 将 默认 值 改 为 特定 的 值 。 


8.1.2 POST 方法 上 传 


在 PHP 系统 中 ,文件 的 上 传 是 通过 HTML 表单 中 的 file 控件 将 文件 上 传 至 php.ini 
文件 的 upload_tmp_dir 选项 所 指定 的 临时 目录 ， 然 后 由 PHP 的 函数 move_uploaded 
_file(0) 将 上 传 的 临时 文件 移动 到 指定 的 位 置 实现 的 。 为 了 实现 在 浏览 器 上 选择 要 上 传 
的 文件 ， 需 要 在 HTML 表单 中 加 入 file 控件 ， 并 且 必 须 指定 表单 的 enctype 属性 为 


单元 八 _ 文 件 上 传 和 异常 处 理 


“multipart/form-data”， 才 可 以 上 传 文件 。 
下 面 的 示例 演示 了 在 客户 端 选择 上 传 文件 的 HTML 表单 的 内 容 ， 代 码 如 下 。 


<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0rg/TR/xhtmll/DTD/xhtml1-transitional.dtd"> 

<html xmlns="http:/www.w3.org/1999/xhtml"> 

<head> 

<meta http-equiv="Content-Type" content="text/html; charset=gb2312"> 

<title> 文 件 上 传 </title> 

</head> 

<body> 

<form method="post" action="uploadFile.php" enctype="multipart/form-data"> 

<input type="hidden" name="MAX FILE SIZE" value="30000" /> 

请 选择 要 上 传 的 文件 <input type="file" id="upfile" name="upfile"/> 

<input type="submit" value=" 上传" id="submit"/> 

</form> 

</body> 

</html> 


程序 运行 结果 如 图 8-1 所 示 。 


TID 


表单 的 action 属性 为 接收 表单 内 容 的 文件 ， MAX_FILE_SIZE 隐藏 字段 必须 存放 
在 file 控件 前 ， 其 值 为 接收 文件 的 最 大 尺寸 ， 单 位 为 字 节 。 当 选择 大 于 此 尺寸 的 文件 
时 ， 浏 览 器 会 拒绝 上 传 。 但 是 在 浏览 器 端 可 以 简单 绕 过 此 限制 ， 不 能 只 依靠 该 值 来 限 
制 上 传 大 文件 。 对 PHP 中 设置 的 最 大 上 传 限 制 是 不 会 失效 的 ,也 避免 了 用 户 浪费 时 间 
等 待 大 文件 上 传 完毕 之 后 才 发 现 文件 尺寸 过 大 上 传 失 败 的 麻烦 。 

在 浏览 器 上 选择 了 要 上 传 的 文件 ， 单 击 “ 上 传 ” 按 钮 ， 所 选择 文件 将 上 传 到 临时 
文件 夹 , 所 上 传 文件 的 信息 将 放 在 全 局 变量 $ FILES 中 。 该 全 局 变量 是 一 个 二 维 数组 ， 
第 一 维 键 名 为 表单 file 控件 的 name 属性 值 ， 第 二 维 包含 如 表 8-2 所 示 的 信息 。 


表 8-2 
描 述 
客户 端 所 选 文件 的 源 文件 名 
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客户 端 所 选 文件 的 类 型 。 如 果 客 户 端的 浏览 器 提供 此 信息 ， 该 项 
才 会 有 值 

所 上 传 文件 的 大 小 ， 单 位 为 字 节 

客户 端 所 选 文件 上 传 至 服务 器 后 所 存储 的 临时 文件 名 
文件 上 传 的 错误 代码 。 上 传 成 功 为 0 


文件 上 传 的 错误 代码 如 表 8-3 所 示 。 


错误 代码 
UPLOAD ERR OK 


UPLOAD ERR INI SIZE 


UPLOAD ERR FORM SIZE 


UPLOAD ERR PARTIAL 
UPLOAD ERR NO FILE 

UPLOAD ERR NO TMP DIR 
UPLOAD ERR CANT WAITE 


表 8-3 
描 述 
其 值 为 0， 表 示 没有 错误 发 生 ， 文 件 上 传 成 功 
其 值 为 1， 表 示 上 传 的 文件 超过 了 php.ini 中 upload max_ 
filesize 选项 限制 的 值 
其 值 为 2, 表示 上 传 文件 的 大 小 超过 一 个 HTML 表单 中 MAX_ 
FILE_SIZE 选项 指定 的 值 
其 值 为 3， 表示 文件 只 有 部 分 被 上 传 
其 值 为 4， 表示 没有 文件 被 上 传 
其 值 为 6， 表示 找 不 到 临时 文件 夹 
其 值 为 7， 表示 文件 写 入 失败 


接收 文件 上 传 的 PHP 脚本 是 为 了 决定 接 下 来 该 对 文件 进行 哪些 操作 , 应 该 实现 四 
辑 上 必要 的 检查 。 例 如 ， 检 查 上 传 的 文件 是 否 超过 限制 ， 然 后 再 根据 错误 代码 进行 下 
一 步 的 限制 。 如 果 上 传 成 功 ， 则 将 其 转移 至 指定 的 目录 ， 和 否则 从 临时 文件 夹 中 删除 上 


传 的 文件 。 


下 面 示例 演示 了 使 用 PHP 脚本 处 理 上 传 的 文件 ， 代 码 如 下 所 示 。 


<2php 


S$uploadPage = "file_test.html"; 
$dir = dirmame(realpath(_ FILE  )) . DIRECTORY SEPARATOR: 
SmaxUploadSize = ini_get(upload max filesize’); 


Serr_msg = false; 


ilisset($_FILES[upfile])){ 
Serr_ masg = "表单 不 完全 ! “ 


} else{ 


SfileImg = $_FILES["upfile"]: 


} 


switch($fileImg['error]){ 
case UPLOAD ERR INI SIZE: 
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S$err msg = "文件 超过 最 大 上 传 限 制 :SmaxUploadSize \n"; 
break; 

case UPLOAD ERR PARTIAL: 
Serr_msg = "文件 上 传 不 完全 .请 重新 <a href="{$uploadPage}'> 上 传 </a>\n"; 
break; 

case UPLOAD ERR NO FILE: 
Serr_msg = "没有 选择 文件 .请 重新 <a href='{$SuploadPage}'> 上 传 </a>\n"; 
break:; 

case UPLOAD ERR FORM SIZE: 
Serr_msg = "文件 超过 页 面 最 大 上 传 限制 ."; 
break:; 

case UPLOAD ERR CANT WRITE: 
Serr_ msg = "文件 写 入 失败 .请 重新 <a hre 人 ='{$uploadPage}'> 上 传 </a>\n"; 
break; 

case UPLOAD ERR NO_TMP DIR: 
Serr msg = "没有 临时 文件 夹 .请 重新 <a hre 仁 '{$uploadPage}'> 上 传 </a>\n"; 
break; 

case UPLOAD ERR OK: 
break: 

default: 


Serr_msg = "未 知 错误 .请 重新 <a hre 人 ='{$uploadPage}'> 上 传 </a>\n"; 


} 
if(in_array($fileImg['type'],array('image/ipeg','image/ipg',image/PNG')){ 
Serr_msg = "只 允许 上 传 .png 或 .jpg 图 片 .请 重新 <a hre 伍 '{$uploadPage}'> 上 传 </a>\n"; 
b 
if(!$err msg){ 
if(!move_uploaded file($fileImg['tmp_name'], $dir.$fileImg['name'])){ 
Serr_msg = "移动 文件 失败 .请 重新 <a hre 伍 '{$uploadPage}'> 上 传 </a>\n"; 
; 
} 
if($err_ msg){ 
echo $err msg; 
Jelse{ 
echo "<img src="{$fileImg["name']}' alt=" 上 传 的 文件 ' title=" 上 传 的 文件 ' />"; 
echo "上 传 成 功 "; 
用 
?> 


文件 上 传 成 功 后 结果 如 图 8-2 所 示 。 
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8-2 
如 果 文 件 过 大 ， 则 会 出 现 如 图 8-3 所 示 的 错误 提示 。 
ET 


如 果 想 在 显示 图 片 时 指定 图 片 的 宽度 和 高 度 ， 可 使 用 imagesize() 函 数 获取 。 


8.1.3 同时 上 传 多 个 文件 


在 实际 应 用 中 , 有 时 需要 批量 上 传 文件 ,采用 前 面 的 方式 一 次 只 能 上 传 一 个 文件 。 
如 何 能 够 批量 上 传 多 个 文件 呢 ? 
先 看 看 下 面 示例 的 HTML 代码 ， 代 码 如 下 所 示 。 


<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.org/TR/xhtmll/DTD/xhtmll-transitional.dtd"> 

<html xmlns="http://www.w3.org/1999/xhtml"> 

<head> 

<meta http-equiv="Content-Type" content="text/html; charset=gb2312"> 

<title> 文 件 上 传 </title> 

</head> 

<body> 

<form method="post" action="ftp.php" enctype="multipart/form-data"> 

请 选择 要 上 传 的 文件 : <br> 

<input type="hidden" name="MAX FILE_SIZE" value="3000000" /> 

<input type="file" id="upfilel" name="upfilel[]"/><br/> 

<input type="hidden" name="MAX FILE SIZE" value="3000000" /> 
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<input type="file" id="upfile2" name="upfilel[]"/><br/> 

<input type="hidden" name="MAX _FILE_SIZE" value="3000000" /> 
<input type="file" id="upfile3" name="upfilel[]"/><br/> 

<input type="hidden" name="MAX FILE SIZE" value="3000000" /> 
<input type="file" id="upfile4" name="upfilel[]"/><br/> 

<input type="hidden" name="MAX FILE SIZE" value="3000000" /> 
<input type="file" id="upfile5" name="upfilel[]"/><br/> 

<input type="submit" value=" 上传" id="submit"/> 

</form> 

</body> 

</html> 


HTML 页 面 效 果 如 图 8-4 所 示 。 


在 上 面 的 程序 中 ， 可 同时 上 传 5 个 文件 。 在 每 一 个 file 控件 前 面 都 要 加 上 一 个 隐 
藏 字段 。 同 时 每 个 file 控件 的 name 属性 均 采 用 HTML 数组 格式 ， 即 在 值 后 面 加 “ 口 ”。 
下 面 是 使 用 PHP 文件 处 理 从 客户 端 同 时 上 传 的 多 个 文件 ， 代 码 如 下 。 


<?php 

SuploadPage = "ftp_test.html": 

$dir = dimame(realpath( FILE )). DIRECTORY _ SEPARATOR: 
$maxUploadSize = ini_get('upload max filesize"): 


S$err_msg = false; 


iflisset($_FILES[upfilel])){ 
S$err_ msg = "表单 不 完全 ! 请 重新 <a href="{$uploadPage}'> 上 传 </a><br>"; 
echo $err mseg; 
exit; 


b 


for($i=0;$1<5;$1++){ 
S$fileImg=$ FILES[upfilel']: 
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switch($fileImg['error][Si){ 
case UPLOAD ERR INI SIZE: 
Serr_msg = $fileImg[mame"][$i] ." 文件 超过 最 大 上 传 限 制 :SmaxUploadSize <br>"; 
break 
case UPLOAD ERR PARTIAL: 
Serr_msg = $fileImg[mame'][$i] ." 文件 上 传 不 完全 .请 重新 <a hre 伍 
"{SuploadPage}'> 上 传 
</a><br>"; 
break; 
case UPLOAD ERR NO FILE: 
Serr_msg = $fileImg["name'][$i] .” 没有 选择 文件 .请 重新 <a href='{$uploadPage}'> 
上 传 
</a><br>"; 
break:; 
case UPLOAD ERR FORM SIZE: 
Serr_msg = $fileImg["name'][$i] ." 文件 超过 页 面 最 大 上 传 限 制 ."; 
break; 
case UPLOAD ERR CANT WRITE: 
Serr_msg = $fileImg['name'][$i] ." 文件 写 入 失败 .请 重新 <a href='{$uploadPage}'> 
上 传 
</a><br>"; 
break; 
case UPLOAD ERR NO_TMP DIR: 
Serr_msg = $fileImg['name"][$i] ." 没有 临时 文件 夹 .请 重新 <a hre 人 '{$uploadPage}'> 
上 传 
</a><br>"; 
break: 
case UPLOAD ERR ORK: 
break:; 
default: 
Serr_msg = $fileImg['name'][$i] .” 未 知 错误 .请 重新 <a hre 伍 '{$uploadPage}> 上 传 
</a><br>"; 
} 
if(!in_array(SfileImg['type'][$i],array('image/ipeg','image/pjpeg',image/png'’){ 
Serr msg 二" 只 允许 上 传 .png 或 jpg 图片 .请 重新 <a href='{$uploadPage}'> 上 传 
</a><br>"; 
} 


if(!$err msg){ 
if(!move_uploaded file($fileImg['tmp_name'][$i], Sdir.$fileImg[mame'][$i)){ 
$err_msg = "移动 文件 失败 $i. 请 重新 <a href='{$uploadPage}'> 上 传 </a><br>"; 
> 
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这 $err msg){ 
echo $err msg; 
yelse{ 
echo "<img src='{$fileImg['name"][$i]}》' alt=' 上 传 的 文件 ' title=' 上 传 的 文件 ' />"; 
echo "上 传 成 功 <br>"; 
F 
Serr_ msg = false; 
} 
?> 


上 传 后 如 果 指 定 的 文件 无 效 ， 则 结果 如 图 8-5 所 示 。 


. jpg 图片 . 请 重新 
.jpg 图片. 请 重新 
.jpg 图片 .请 重新 
. jpg 图 片 . 请 重新 ， 
, jpg 图 片 . 请 重新 


这 个 程序 中 ， 脚 本 文件 夹 接收 从 HIML 表单 上 传 的 文件 ， 并 采用 循环 的 方式 检查 
一 个 上 传 的 文件 ， 检 测 通过 则 显示 该 图 片 ， 否 则 显示 错误 信息 。 


在 批量 上 传 文件 时 ,HTML 表单 中 file 控件 的 name 属性 一 定 要 采用 ML j ' 
数组 形式 ， 这 样 在 服务 器 端 才 方便 对 上 传 的 文件 进行 处 理 。 


8.1.4 文件 的 下 载 


在 实际 应 用 中 ，Web 服务 器 通常 会 提供 文件 下 载 功能 。 对 于 文件 下 载 ， 只 需要 给 


出 要 下 载 文件 所 在 的 位 置 即 可 。 
下 面 的 示例 演示 了 通过 PHP 进行 文件 的 下 载 ， 代 码 如 下 所 示 。 


<?php 
Sdowndir ="."; 
SdirHandle = @opendir(Sdowndir): 
while(Sfilen = readdir(SdirHandle)) { 
if(Sfilen <> "." && $filen <> "..") 
echo "<a href-'$filen'>$filen</a><br>"; 
} 
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closedir($dirHandle); 
Ms 


程序 运行 结果 如 图 8-6 所 示 。 


downloadfile. php 
file test. html 
Etp. php 
ftp_test. html 
uploadFile. php 


图 8-6 


在 这 个 程序 中 ， 采 用 遍历 给 定 文件 夹 的 方式 将 下 载 文件 夹 中 的 文件 读 出 ， 然 后 再 
给 出 每 个 文件 的 链接 地 址 。 


8.2 PHP 错误 类 型 


在 程序 开发 中 ， 难 免 会 因为 某 种 原因 而 产生 错误 。 如 何 去 避 免 、 调 试 、 修 复 错误 
并 对 程序 可 能 发 生 的 异常 进行 处 理 是 一 个 程序 员 必 备 的 能 力 。PHP 提供 了 良好 的 错误 
提示 ， 在 进行 程序 调试 时 可 根据 提示 信息 对 错误 进行 排除 。 
在 PHP 程序 开发 中 ， 通 常会 出 现 以 下 5 种 错误 。 
语法 错误 : 在 程序 中 使 用 了 错误 的 语法 而 导致 的 错误 。 
语义 错误 : 在 程序 中 正确 地 使 用 了 PHP 的 语法 ， 但 是 没有 任何 意义 ， 程 序 达 
不 到 预想 的 效果 。 
逻辑 错误 : 在 程序 中 使 用 的 逻辑 与 实际 上 需要 的 逻辑 不 符 。 
注释 错误 : 在 程序 中 写 的 注释 与 该 程序 代码 的 意义 不 符 。 
运行 错误 : 由 于 运行 环境 等 原因 而 导致 的 错误 。 
在 以 上 几 种 错误 中 ,， 除 最 后 一 种 是 由 于 PHP 所 运行 的 环境 原因 等 造成 的 以 外 ,前 
面 4 种 均 是 由 程序 开发 人 员 造 成 的 ， 因 而 这 4 种 错误 应 该 在 程序 开发 中 尽量 避免 。 


8.2.1 语法 错误 


在 程序 中 使 用 了 错误 的 语法 ， 会 产生 一 个 语法 错误 。 
下 面 是 语法 错误 的 示例 。 


<?php 
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Snumber = 80; 
Sprice = 5.5; 
$sum = Snumber *" $price: /使 用 了 错误 的 语法 
echo $sum 
ye 
程序 运行 结果 如 图 8-7 所 示 。 


EB og /Tes 


该 错误 为 解析 错误 ， 有 具体 在 脚本 文件 的 第 4 行 ， 用 户 看 到 该 错误 时 ， 可 直接 检查 
该 行程 序 ， 一 般 情 况 下 ， 错 误 就 发 生 在 该 行 。 


8.2.2 语义 错误 


语义 错误 是 在 使 用 了 正确 语法 的 基础 上 ， 使 用 了 错误 的 格式 而 导致 的 。 
下 面 示例 演示 了 语义 错误 的 示例 ， 代 码 如 下 。 

<?php 

$sl = "中 国 ": 

$s2 = "湖北 武汉 ": 

$s3 = $sl + $s2; /这 里 使 用 了 错误 的 字符 串 连接 符 

echo $s3 

> 


在 上 面 的 程序 中 ， 错 误 地 使 用 了 “+” 作 为 字符 串 连 接 符 ， 因 为 PHP 能 够 自动 进 
行 隐 式 变量 类 型 转换 ，PHP 在 解析 时 认为 它 是 符合 PHP 的 语法 的 ， 并 不 会 提示 出 错 。 


8.2.3 ”逻辑 错误 


逻辑 错误 对 于 PHP 来 讲 不 是 错误 ， 因 为 语法 、 语 义 上 没有 任何 问题 , 但 是 因为 程 
序 代码 存在 着 逻辑 的 问题 ， 进 而 导致 程序 得 不 到 所 期 望 的 结果 。 
下 面 示例 演示 了 逻辑 错误 的 示例 ， 代 码 如 下 。 


<2php 
Sage = 10; 
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if($age > 18){  // 浏 断 年 龄 
echo "未 成 年 1"; 
Jelse{ 
echo "已 成 年 "; 
} 
?> 


在 上 述 程序 中 ， 程 序 本 身 从 语法 上 和 语义 上 讲 都 没有 任何 问题 ， 并 且 能 够 得 到 结 
果 ， 但 是 程序 逻辑 有 问题 ， 本 来 年 龄 小 于 18 岁 为 未 成 年 ， 但 是 程序 却 将 大 于 18 岁 认 
为 是 未 成 年 ， 这 就 产生 了 逻辑 错误 。 这 种 错误 在 开发 时 应 尽量 避免 。 


8.2.4 注释 错误 


注释 对 于 程序 来 讲 是 必 不 可 少 的 。 因 为 在 分 布 式 开发 中 ， 随 时 都 有 可 能 去 读 其 他 
程序 员 编写 的 代码 ， 如 果 没 有 注释 ， 将 会 花费 大 量 的 时 间 去 读 懂 别 人 的 代码 。 另 外 ， 
代码 没有 注释 ， 后 期 的 维护 也 是 相当 困难 的 。 对 于 注释 错误 ， 比 没有 注释 更 加 可 怕 ， 
因为 开发 人 员 往 往 会 只 看 注释 不 会 再 花 时 间 去 看 代码 。 

下 面 示例 演示 了 一 个 注释 错误 的 示例 ， 代 码 如 下 。 


<?php 

$age= 10; 

if($age < 18){ /判断 年 龄 是 否 大 于 18 
echo "未 成 年 1"; 

jelse{ 

echo "已 成 年 "; 

} 

?> 


上 面 程序 中 ， 显 示 注 释 和 程序 本 身 的 逻辑 不 统一 ， 虽 然 注 释 错误 对 于 程序 本 没有 
任何 影响 ， 但 是 却 影响 以 后 对 代码 的 维护 与 修改 。 


8.2.5 运行 错误 


行 错误 与 程序 代码 无 关 ， 它 是 由 脚本 运行 的 环境 等 因素 造成 的 。 
面 示例 演示 了 运行 错误 ， 代 码 如 下 。 


<?php 

class Book { 

Private $bookId; 

Private $bookName; 

public function __construct($bookId,$bookName){ 
S$this->bookId=$bookId; 
Sthis->bookName=$bookName; 


运 
下 
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于 


public function getBookId0O { 
return $this->bookId; 
} 


public function getBookName() { 
Teturn $this->bookName; 

} 

} 
$bookl = new Book(); 
echo "书本 编号 : ".$book1->getBookId(."<br>"; 
echo "书本 名 称 : ".$book1->getBookName(); 
?> 


上 述 程序 中 ， 创 建 一 个 对 象 book1， 但 是 没有 给 bookId 和 bookName 赋值 ， 所 以 
调用 时 会 产生 错误 。 


8.3 PHP 错误 处 理 


不 管 是 程序 引发 的 错误 ， 还 是 环境 因素 引发 的 错误 ， 默 认 情况 下 ，PHP 都 会 给 出 
提示 信息 。 这 些 提示 信息 包含 服务 器 的 运行 环境 信息 。 在 实际 的 Web 环境 中 ,将 这 些 
信息 显示 出 来 ， 必 然 给 服务 器 带 来 安全 隐患 。 因 此 ， 必 须 对 可 能 出 现 的 错误 进行 相 
应 的 处 理 。 


8.3.1 ”错误 级 别 


PHP 中 的 错误 是 通过 一 个 错误 级 别 来 进行 划分 的 。 从 最 基本 的 通告 到 最 严重 的 错 
误 ， 错 误 级 别 标识 着 所 产生 的 错误 的 严重 性 。 表 8-4 所 示 列 出 了 错误 级 别 的 种 类 。 


表 8-4 


这 是 一 个 严重 错误 ， 不 可 恢复 ， 如 位 置 异 常 、 内 存 不 足 等 
E WARNING 警告 ， 最 一 般 的 错误 ， 如 函数 的 参数 错误 等 
E PARSE 解析 错误 ， 在 解析 PHP 文件 时 产生 ， 并 强制 PHP 在 执行 前 退出 
这 个 错误 级 别 是 唯一 不 包含 在 E_ALL 常量 中 的 ， 主 要 是 为 了 便于 兼容 
PHP 的 低 版 本 
E NOTICE 通常 表示 可 能 在 操作 一 些 未 知 的 变量 等 
这 个 内 部 错误 是 由 于 PHP 加 载 扩 展 失 败 而 导致 的 , 并 且 会 导致 PHP 停止 运 
行 并 退出 


E_STRICT 


FE_CORE ERROR 
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E_COMPILE ERROR | 编译 错误 是 在 编译 时 发 生 的， 这 个 错误 将 导致 PHP 运行 退出 

E COMPILE WARNING | 编译 警告 用 于 告诉 用 户 一 些 不 推荐 的 语法 信息 

用 户 定义 的 错误 将 导致 PHP 退出 执行 。 它 不 是 来 自 PHP 本 身 ， 而 是 来 自 脚 
本 文件 中 

E_USER WARNING | 脚本 使 用 它 来 通知 一 个 执行 失败 ， 同 时 PHP 也 会 用 E_WARNING 通知 
E USER NOTICE 用 户 定义 的 通告 用 于 在 脚本 中 表示 可 能 发 生 的 错误 


E_USER ERROR 


8.3.2 ”php.ini 对 错误 处 理 的 设置 


PHP 的 环境 几乎 都 是 在 php.ini 文件 中 进行 设置 ， 有 两 项 关于 错误 处 理 的 设置 : 一 是 
display_errors， 另 一 个 是 error reporting。 前 一 个 变量 用 来 告诉 PHP 是否 显示 错误 ， 它 的 默 
认 值 是 off， 即 不 显示 错误 信息 ， 如 果 设 置 为 tue， 将 显示 错误 信息 。 后 一 个 变量 是 告知 
PHP 如 何 显示 提示 信息 ， 默 认 值 为 E ALL&~E _NOTICE， 即 显示 除 提示 信息 外 的 所 有 
错误 信息 。 

对 于 开发 环境 ， 为 了 便于 程序 的 调试 ， 可 将 php.ini 的 相应 项 改 成 如 下 值 。 


display_errors = on 
error_ reporting =E ALL&~E NOTICE 


对 于 实际 的 Web 环境 ， 可 将 php.ini 的 相应 项 设 为 如 下 值 。 
display_errors = off 
error reporting =E_ALL 


修改 以 后 ， 重 启 apache 服务 器 即 可 。 除 了 可 在 php.ini 文件 进行 设置 以 外 ， 还 可 
以 在 程序 开头 使 用 error_reporting() 函 数 进行 设置 。 该 函数 的 参数 与 php.ini 文件 中 的 
一 样 。 


8.3.3 ”错误 处 理 


在 程序 中 ， 因 各 种 原因 可 能 导致 的 错误 ，PHP 都 会 给 出 相应 的 提示 信息 。 而 对 于 
错误 信息 的 处 理 , 除了 可 采用 在 php.ini 文件 中 进行 设置 的 方法 外 ,还 可 以 直接 在 程序 
中 进行 设置 。 

1. 错误 信息 的 隐藏 

对 于 将 整个 系统 可 能 产生 的 错误 信息 进行 隐藏 ， 可 采用 error_reporting() 函 数 进行 
设置 。 下 面 示例 演示 了 如 何 隐 藏 所 有 错误 信息 ， 代 码 如 下 所 示 。 


<?php 
error reporting(0): /使 用 函数 error_ reporting0 隐 藏 所 有 提示 信息 
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class Book { 
Private $bookId; 
Private SbookName: 
public function _construct($bookId,$bookName){ 
S$this->bookId=$bookId; 
S$this->bookName=$bookName:; 
; 


public function getBookId0O { 
return $this->bookId; 
} 


public function getBookName() { 
Teturn $this->bookName; 
} 
echo "书本 编号 : ".Sbookl->getBookId0."<br>": 
echo "书本 名 称 : ".$Sbook1->getBookName(); 
?> 


上 面 的 代码 中 ， 使 用 error reporting(0) 函 数 隐藏 了 所 有 提示 信息 ， 因 此 即使 程序 


中 产生 错误 ， 也 不 会 显示 出 错 信息 。 显 示 效 果 大 家 可 以 自行 运行 看 一 看 。 

对 于 单条 语句 可 能 产生 的 错误 信息 的 隐藏 ， 可 采用 在 语句 前 加 “@” 符 号 进行 隐 
藏 。 下 面 修改 上 例 代 码 演示 采用 “@” 符 号 隐藏 单 条 语句 可 能 产生 的 错误 ， 代 码 如 下 
所 示 。 


<2php 

class Book { 

Private $bookId: 

Private $bookName: 

public function __construct($bookId,$bookName){ 
S$this->bookId=$bookId; 
S$this->bookName=$bookName; 

} 


public function getBookId() { 
return $this->bookId: 


public function getBookNameO { 
Teturn $this->bookName; 
} 
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echo "书本 编号 : ".@Sbookl->getBookId0."<br>"; /使 用 @ 符 号 隐藏 可 能 产生 的 错误 信息 
echo "书本 名 称 : ".@S$bookl->getBookName0: 
?> 


2. 超时 错误 的 处 理 


对 于 Web 系统 而 言 ， 很 多 时 候 可 能 因为 网 速 太 慢 或 者 进行 了 过 多 的 操作 ， 处 理 时 间 超 
过 了 apache 服务 器 所 设置 的 最 长 超时 时 间 。 此 时 apache 服务 器 将 中 断 连 接 , 并 抛 出 类 似 如 
图 8-8 所 示 的 错误 信息 。 


Fatal error: Jiaximum execution time of 30 seconds exceeded 
in C:\wamp\www\Test\timeout. php on line 3 


出 现 了 超时 错误 后 ， 其 后 的 代码 将 不 会 被 执行 。 因此， 可 以 在 一 些 可 能 会 超时 的 
脚本 中 进行 处 理 ， 让 程序 能 够 正常 运行 。 通 常 采用 的 方法 就 是 采用 set_time_limit() 函 
数 来 延长 脚本 的 运行 时 间 。 其 语法 格式 如 下 所 示 。 


Void set_time_limit(int $seconds) 


其 中 ， 参 数 seconds 为 要 延长 的 秒 数 。 该 函数 没有 返回 值 。 
下 面 示例 演示 了 set_time limit 函数 的 应 用 ， 代 码 如 下 所 示 。 


<?php 
set_time_limit(]): 
for(G$i=0::$i++){ 
if($i % 50000 = 0){ 
echo $i."<br>"; 
set_ time limit(1); 
} 
if($i >= 5000000) exitO; 
} 
?> 


在 上 面 的 程序 中 ， 用 户 可 自行 将 第 二 个 set_time limit() 函 数 去 掉 ， 然 后 运行 程序 ， 
就 会 出 现 前 面 所 讲 的 超时 错误 信息 。 
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8.4 PHP 错误 处 理 


在 PHP 中 ， 异 常 处 理 是 使 用 关键 字 try、catch 和 throw 来 实现 的 。 将 需要 进行 异 
常 处 理 的 代码 放 入 try 代码 块 内 ， 以 便 捕获 可 能 存在 的 异常 。 每 一 个 try 语句 必须 至 少 
有 一 个 catch 语句 与 之 对 应 ，catch 语句 用 于 捕获 异常 。 使 用 多 个 catch 语句 可 以 捕获 
不 同类 所 产生 的 异常 。throw 用 于 抛 出 异常 。 语 法 如 下 。 


Ty 


Throw new Exception($error); 
}catch(FirstException $exception){ 


}catch(SecondException $exception){ 


} 


其 中 ， 在 try 和 catch 之 间 放 置 可 能 发 生 异 常 的 代码 ， 当 检测 到 异常 时 ， 便 使 用 throw 
关键 字 抛 出 异常 ，catch 语句 用 于 捕获 所 抛 出 的 异常 。 可 以 看 出 ， 每 一 个 try 语句 至 少 
有 一 个 catch 语句 与 之 对 应 。 实 际 上 ， 在 应 用 中 可 以 使 用 多 个 catch 关键 字 与 之 对 应 ， 
以 捕获 不 同类 所 引起 的 异常 。 多 个 catch 关键 字 会 按 顺 序 执行 ， 直 到 所 有 异常 捕获 
完成 。 

Throw 语句 只 能 够 抛 出 对 象 。 它 不 能 抛 出 任何 其 他 的 基础 数据 类 型 ， 如 字符 串 或 数 
组 。 PHP 内 置 了 一 个 异常 处 理 类 Exception, 所 有 自己 使 用 的 异常 类 都 必须 从 它 继承 出 来 。 
如 果 尝 试 扫 出 一 个 不 是 由 Exception 类 继承 的 类 ， 将 得 到 一 个 运行 错误 。 


1. 异常 类 
异常 类 是 PHP 内 置 的 异常 处 理 类 ， 该 类 的 定义 如 下 。 


<?php 

class Exception{ 
protected $message = Unknown exception': ”// 异 常 信息 
protected Scode = 0; /用 户 自 定义 异常 代码 
protected $file; /发 生 异 常 的 文件 名 
Protected $line; /发 生 异 常 的 代码 行 号 


fonction _construct($message = null , $code = 0); 
final function getMessage(); // 返 回 异 常 信息 


final function getCode(); // 返 回 异 常 代码 
final function getFile(); // 返 回 发 生 异 常 的 文件 名 
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final function getLine():; // 返 回 发 生 异 常 的 代码 行 号 

final function getTrace(); /backtraceO 数 组 

final function getTraceAsString(); // 已 格式 化 成 字符 串 的 getTraceO 信 息 
function _toStringO; // 可 重 载 的 方法 和 输出 的 字符 串 


} 
?> 


异常 处 理 类 用 于 在 脚本 发 生 异 常 时 建立 异常 对 象 ， 该 对 象 将 用 于 存储 异常 信息 并 
用 于 抛 出 和 捕获 。 在 准备 抛 出 异常 时 ， 需 要 建立 异常 对 象 。 其 语法 格式 如 下 。 


Sexcept = new Exception([string $errmsg[,int $errcode]]); 


其 中 ， 参数 errmsg 为 用 户 自 定义 的 错误 信息 ，errcode 为 表示 用 户 自 定义 的 异常 代码 。 
下 面 示例 利用 异常 类 进行 异常 处 理 ， 代 码 如 下 。 


<?php 
class NullHandleException extends Exception 
{ 
function _ construct($msg) 
{ 
parent::_ construct($msg); 
} 
} 
function printObj($obj) 
dl 
这 $obj =— nulD){ 
throw new NullHandleException(" 空 对 象 ."); 
} 
Print $obj ." 
1 
class Student{ 
Private $name; 
function construct($name) 
{ 
$this->name = $name; 
} 
function _ toString() 
4 
Teturn $this->name; 
} 
上 
try{ 
printObj(new Student(" 张 明 ")); // 调 用 函数 
PrintObj(NULL): 
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printObj(new Student(" 王 二 小 ")); 

} catch(NullHandleException $exception){ // 捕 获 异常 

echo $exception->getMessage(); 

} catch(Exception $exception){ // 这 里 没有 任何 语句 ， 不 会 执行 
} 
?> 


程序 代码 运行 结果 如 图 8-9 所 示 。 


2. 设置 异常 处 理 函 数 set_exception_handler() 


在 实际 的 应 用 中 ， 除 了 捕获 可 能 发 生 的 异常 外 ， 还 可 以 设置 一 个 函数 来 处 理 一 些 
可 能 没有 被 获取 的 异常 。PHP 提供 了 set_exception handler() 函 数 设置 异常 处 理 函 数 ， 


其 语法 格式 如 下 。 
string set_exception handler(callback $exception handler) 


其 中 ， 参 数 exception handler 是 用 于 处 理 未 被 捕获 异常 的 函数 名 。 函 数 设置 成 功 时 返 
回 先前 定义 的 异常 处 理 函 数 名 ， 失 败 时 返回 null， 如 果 未 定义 用 于 捕获 异常 的 函数 ， 
也 会 返回 null。 

用 于 处 理 未 被 捕获 的 异常 函数 的 语法 格式 如 下 所 示 。 


function exception_handler($exception){ ”// 参 数 exception 为 异常 对 象 
} 


下 面 示例 演示 了 设置 异常 处 理 函 数 ， 代 码 如 下 所 示 。 


<?php 

class ExceptionTest { 

public function constructO { 
@set_exception handler(array($this, 'exception_handler"));// 设 置 异 常 处 理 函 数 
throw new Exception(” CLASS _):// 使 用 魔术 常量 CLASS ， 即 抛 出 异常 的 消息 为 类 

名 称 

} 

public function exception handler($exception) { // 定 义 异 常 处 理 函 数 
print "异常 信息 : ". $exception->getMessage( ."<br>"; 
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print "异常 代码 : " . $exception->getCode() . "<br>"; 
print "文件 名 : " . Sexception->getFile() . "<br>"; 
print "异常 所 在 行 : " . Sexception->getLine() . "<br>"; 
print "追踪 路 线 : ";print_r($exception->getTrace());; 
Print $exception->getTraceAsString() . "<br>"; 


} 
Se = new ExceptionTest; 
?> 


在 上 述 程序 中 ， 用 户 自 定义 异常 处 理 函数 中 展示 了 完整 的 异常 信息 。 在 实际 的 系 
统 开发 中 ， 可 采用 此 种 方式 显示 所 有 异常 信息 ， 对 于 程序 的 调试 大 有 益处 。 


8.5 PHP 程序 的 调试 
在 PHP 开发 中 ， 不 可 避免 地 会 对 程序 进行 调试 。 要 进行 调试 ， 首 先 必须 打开 错误 报 
告 , 再 由 相应 的 语句 输出 信息 至 屏幕 , 并 根据 这 些 信息 找 出 并 修复 程序 中 可 能 存在 的 错误 。 
1. 使 用 ECHO 进行 调试 
在 开发 中 ， 当 程序 出 现 警告 或 错误 时 ， 打 开 错误 报告 查看 错误 信息 是 很 好 的 调试 方 


式 。 但 是 当 程序 没有 语法 错误 时 ， 程 序 由 于 某 种 原因 而 得 不 到 期 望 的 结果 ， 采 用 查看 错 
误 报告 的 方式 就 不 能 进行 调试 了 。 此 时 需要 查看 中 间 变 量 的 值 ， 一 步 步 进行 跟 踪 调试 。 
下 面 示例 通过 查看 中 间 变 量 的 值 进行 调试 ， 代 码 如 下 。 


<?php 

Snum 1 = 109; 

echo '$num 1=". $num 1."<br>"; 
Snum 2 = 300; 

echo '$num 2 = ' . $num 2."<br>"; 
$sum = $num 1 + $num 2; 

echo '100 + 300 =". $sum: 

?> 


在 上 面 的 程序 中 ， 采 用 逐 行 输出 变量 值 的 方式 进行 调试 。 通 过 程序 输出 内 容 ， 可 
以 很 容易 地 看 出 程序 的 错误 在 第 一 如 图 8-10 所 示 。 
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这 个 程序 很 简单 ， 几 乎 可 以 直接 看 出 程序 错 在 哪里 。 但 如 果 是 逻辑 复杂 的 程序 ， 
就 不 容易 直接 看 出 ， 也 不 可 能 一 步 步 输出 变量 值 进行 追踪 调试 ， 这 时 可 采用 下 面 的 方 
法 进行 调试 。 

2. 使 用 die() 进 行 调试 


对 于 一 些 逻 辑 相 对 简单 的 程序 ， 可 首先 粗略 地 判断 错误 出 在 哪 一 段 ， 再 从 该 段 输 
出 相关 变量 的 值 ， 根 据 输出 的 变量 值 一 步 步 找 出 程序 中 存在 的 问题 。 但 是 对 于 一 些 好 
辑 复杂 的 程序 而 言 ， 采 用 这 种 方法 不 足以 满足 要 求 ， 此 时 可 使 用 die0 语 句 进 行 调试 。 
die() 语 句 会 中 止 程序 执行 ， 并 在 浏览 器 上 显示 文本 。 如 果 不 想 注释 掉 代 码 ， 只 想 显 示 
到 出 错 之 前 的 信息 和 出 错 信 息 ， 那 么 die0 语 句 特别 有 用 。 

下 面 示例 使 用 了 dieO 进 行程 序 的 调试 ， 代 码 如 下 。 


<?php 
$num = 100; 
Smod = 5; 
for ($i= 0; $i < $num; $1++){ 
if ($i% $mod == 3) { 
die("Error on"."File:". FILE ."online:"._ LINE ); 


alhost/Test/dietest-php - Mi 


Error on File: C:\wamp\wm\Test\dietest. php on line: 6 


阿 Fig 


在 上 面 程序 中 ， 使 用 dieO 语 名 对 程序 进行 调试 。 当 发 生 错误 时 ， 程 序 将 在 错误 
发 生 处 中 止 执行 ， 并 在 浏览 器 显示 出 错 信息 。 根 据 所 提示 的 错误 信息 就 可 以 快速 地 
定位 于 错误 行 ， 并 进行 修改 。 


8.6 使 用 ZendStudio 讲 行 调 试 


前 面 讲 到 了 直接 在 程序 中 输出 中 间 变 量 的 值 或 输出 错误 信息 的 方法 进行 PHP 程序 
的 调试 。 其 实 除 了 这 种 直接 的 方法 以 外 ， 还 可 以 借助 于 第 三 方 的 工具 进行 调试 。 如 
PHPEclipse、ZendStudio 等 专业 的 IDE 开发 工具 ， 都 具有 实时 调试 的 功能 。 
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在 ZendStudio for Eclipse 中 进行 调试 的 步骤 如 下 。 

(1) 在 需要 进行 调试 的 地 方 设 置 断 点 。 设 置 断 点 是 通过 将 光标 移动 到 要 设置 断 点 
的 行 , 再 选择 菜单 中 的 mn 一 Toggle Line Breakpoint 命令 实现 , 也 可 以 双击 要 设置 断 点 
的 行 左 侧 。 设 置 后 该 行 左 侧 将 出 现 一 个 蓝 色 小 圆 点 。 

(2) 选择 菜单 栏 中 的 mn 一 Debug 命令 开始 进行 调试 。 

(3) 在 “调试 ”选项 卡 下 可 以 看 到 进程 在 设置 了 断 点 的 行 时 就 暂停 了 。 此 时 可 以 
通过 单 步 执行 进行 程序 的 调试 。 

在 “Browser Output” 和 “Debug Output” 选 项 卡 下 可 以 实时 看 到 程序 的 输出 结果 ， 
图 8-12 为 Browser Output 选项 卡 显 示 内 容 ， 图 8-13 是 Debug Output 的 值 的 实时 跟踪 


记 Clalei cbr /910 | 50 | 100 | Chr /rray 
[eobol] => 10 
[Bi11] => 20 
a] => 100 


图 8-12 


【单元 小 结 】 


e PHP 可 以 进行 单 文件 和 多 文件 上 传 
e PHP 中 的 异常 类 型 及 处 理 方式 


【单元 自 测 】 


1. 以 下 哪个 比较 将 返回 true? ( 双 选 )(  ) 
A. "ltopg = B. top == 
C.'top' ==0 D23== 23 
- 如 果 用 + 操作 符 把 一 个 字符 串 和 一 个 整 型 数字 相 加 ， 结 果 将 怎样 ?( ) 
A. 解释 器 输出 一 个 类 型 错误 
B. 字符 串 将 被 转换 成 数字 ， 再 与 整 型 数字 相 加 
C. 字符 串 将 被 丢弃 ， 只 保留 整 型 数字 
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D. 字符 串 和 整 型 数字 将 连接 成 一 个 新 字符 串 
3. 考虑 如 下 脚本 。 假 设 http://www.php.net 能 被 访问 ， 脚 本 将 输出 什么 ? (  ) 


<?php 

$s = file_get_contents ("http://www.php.net"); 
strip_tags ($s, array ('p')); 

echo count ($s); 

x 


A. www.php.net 的 主页 的 字符 数 

B. 剔除 标签 后 的 www.php.net 主页 的 字符 数 

| 

D.0 

4. 在 不 把 文件 内 容 预 加 载 到 变量 中 的 前 提 下 ， 如 何 解析 一 个 以 特殊 格式 格式 化 过 的 

多 行文 件 ? (  ) 

A. 用 file0 函 数 把 它 分 割 放 入 数组 B. 用 sscanf() 

C. 用 fscanf() D. 用 fgets() 


【上 机 实战 】 


上 机 目标 


e 练习 将 文件 上 传 
e PHP 中 的 异常 类 型 及 处 理 方法 


上 机 练习 


9 第 一 阶段 多 


练习 : 练习 实现 文件 的 上 传 功能 。 


【问题 描述 】 


在 实际 应 用 中 ， 有 时 需要 用 户 从 本 地 上 传 文件 至 服务 器 进行 处 理 ， 下 面 的 示例 将 
练习 上 传 照片 到 服务 器 。 
编写 代码 实现 如 图 8-14 所 示 的 上 传 功能 。 
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HTML 参考 代码 如 下 。 


<!DOCTYPE html PUBLIC "-WW3CWDTD XHTML 1.0 Transitional//EN" 
"http:Wwww.w3.org/TR/xhtmll/DTD/xhtmll-transitional.dtd"> 
<html xmlns="http://www.w3.org/1999/xhtml"> 
<head> 
<meta http-equiv="Content-Type" content="text/html; charset=gb2312"> 
<title> 文 件 上 传 </title> 
</head> 
<body> 
<form method="post" action="upload.php" enctype="multipart/form-data"> 
<table align="center"> 
<tr> 
<td> 用 户 名 : </td> 
<td><input type="text" name="name"></td> 
</tr> 
<tr> 
<td> 密 码 : </td> 
<td><input type="password" name="pwd"></td> 
</tr> 
<tr> 
<td> 请 选择 照片 :</td> 
<td><input type="hidden" name="MAX FILE SIZE" value="30000" /> 
<input type="file" id="upfile" name="upfile"/> 
<input type="submit" value=" 上传" id="submit"/></td> 
</tr> 
</table> 
</form> 
</body> 
</html> 


单元 八 _ 文 件 上 传 和 异常 处 理 


令 第 二 阶段 多 


练习 : 扩展 的 异常 处 理 类 。 


【问题 分 析 】 


(1) 通常 ， 在 应 用 中 可 能 存在 着 多 种 异常 ， 如 果 所 有 异常 都 使 用 同一 个 异常 类 或 
者 异常 处 理 函 数 去 获取 异常 ， 将 不 利于 问题 的 解决 。 为 此 ， 通 常 是 根据 异常 的 类 型 不 
同 而 定义 不 同 的 自 定义 异常 处 理 类 .这些 自 定义 异常 处 理 类 都 继承 自 PHP 自 带 的 异常 
类 Exception 。 

(2) 下 面 的 程序 中 ， 定 义 了 两 个 文件 操作 类 ， 并 于 try 语句 内 调用 文件 检测 函数 
进行 检测 ， 在 catch 语句 中 抛 出 可 能 发 生 的 异常 ， 并 针对 不 同 的 异常 获取 不 同 的 异常 
信息 。 

程序 代码 如 下 。 


<?php 
class FileExistsException extends Exception{} 
class FileOpenException extends Exception{} 
function openFile($file) 
if(!file_exists($file)) 
Ud 
throw new FileExistsException(" 文 件 不 存在 ! 请 确认 文件 是 否 存 在 ! \n",1); 
} 
if(!fopen($file,"r")) 
{ 
throw new FileOpenException(" 文 件 无 法 打开 ! 请 确认 文件 是 否 有 可 读 权限 ! \n",2); 


openFile(Sfile); 

}catch(FileExistsException $exception){ 

echo "程序 异常 : " . Sexception->getMessage0: 
}catch(FileOpenException $exception){ 

echo "程序 异常 : " . $exception->getMessage(); 
}catch(Exception Sexception){ 

print "异常 信息 : ". $exception->getMessage() ."m": 
print "异常 代码 : ". $exception->getCode( . "\n"; 
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print "文件 名 : " . $exception->getFile( ."\n"; 

print "异常 所 在 行 : " . $exception->getLine() . "\n"; 
print "追踪 路 线 : ";print_r($exception->getTrace()):; 
Print $exception->getTraceAsString() . "\n"; 

} 

?> 


程序 运行 结果 如 图 8-15 所 示 。 


【拓展 作业 】 


1. 写 一 段 代 码 ， 读 取 当 前 目录 下 的 update 目录 中 的 所 有 文件 并 显示 。 
2. 写 一 段 代 码 ， 调 用 系统 函数 ， 显 示 当 前 页 面 的 所 有 错误 。 
3. 写 一 段 代 码 ， 使 用 特殊 字符 ， 屏 蔽 当前 行 可 能 出 现 的 错误 。 
4. 请 描述 一 下 ， 使 用 PHP 进行 文件 上 传 和 下 载 的 关键 步骤 和 代码 。 
文件 上 传 : 
关键 步骤 一 : 
关键 步骤 二 : 
关键 步骤 三 : 
关键 步骤 四 : 
关键 步骤 五 : 
关键 代码 一 : 
关键 代码 二 : 
关键 代码 三 : 
关键 代码 四 : 
关键 代码 五 : 
文件 下 载 : 
关键 步骤 一 : 
关键 步骤 二 : 
关键 步骤 三 : 
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PHP 标 作 MySOQL 
数据 库 


课程 目标 


PP 了 解 PHP 连接 MySQL 数据 库 的 方法 
P 掌握 PHP 对 MySQL 中 的 数据 进行 操作 的 方法 
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Pun 


在 用 户 能 够 访问 并 处 理 数据 库 中 的 数据 之 前 , 必须 创建 到 达 数据 库 的 连接 .在 PHP 
中 ， 这 个 任务 通过 mysql_connect() 函 数 完成 。 脚 本 一 结束 ， 就 会 关闭 连接 。 如 需 提 前 
关闭 连接 ， 请 使 用 mysql_ close() 函 数 。 本 单元 重点 介绍 如 何 连接 数据 库 和 PHP 在 
MySQL 中 实现 数据 操作 。 


9.1 PHP 访问 MySQL 数据 库 


在 PHP 中 ,支持 对 多 种 数据 库 的 操作 ， 且 提供 了 相关 的 数据 库 连 接 函 数 或 操作 函 
数 。 特 别 是 PHP 与 MySQL 数据 库 的 组 合 ，PHP 提供 了 强大 的 数据 库 操作 函数 ， 读 者 
可 直接 在 PHP 中 使 用 这 些 函 数 进行 数据 库 的 操作 。 与 原来 学 过 的 Java、C# 访 问 数据 
库 一 样 ， 操 作 数据 库 ， 首 先 需要 进行 数据 库 的 连接 ， 然 后 选择 需要 进行 操作 的 数据 库 ， 
再 执行 相关 的 数据 库 操作 ， 最 后 需要 关闭 所 建立 的 数据 库 连 接 。 下 面 对 在 PHP 中 如 何 
进行 数据 库 的 操作 进行 详细 的 讲解 。 


9.1.1 连接 MySQL 数据 库 


在 PHP 中 ， 要 对 数据 库 进 行 操 作 ， 首 先 需 要 连接 数据 库 。 连 接 数 据 库 可 使 用 
mysql_connect() 函 数 。 

mysql_connect() 函 数 的 定义 和 用 法 如 下 。 

e mysql connectO 函 数 打开 非 持 久 的 MySQL 连接 。 

语法 如 下 。 


mysql_connect(server,user,pwd,newlink,clientflag) 
其 中 ， 各 参数 说 明 如 表 9-1 所 示 。 


可 选 。 规 定 要 连接 的 服务 器 。 可 以 包括 端口 号 ， 例 如 “hostname:port”， 或 者 到 
本 地 套 接 字 的 路 径 ， 例 如 对 于 localhost 的 “:/path/to/socket”。 如 果 PHP 指令 
Imysql.default host 未 定义 (默认 情况 )， 则 默认 值 是 'localhost:3306' 

可 选 。 用 户 名 。 默 认 值 是 服务 器 进程 所 有 者 的 用 户 名 

可 选 。 密 码 。 默 认 值 是 空 密码 

可 选 。 如 果 用 同样 的 参数 第 二 次 调用 mysql_ connect0， 将 不 会 建立 新 连接 ， 而 
将 返回 已 经 打开 的 连接 标识 。 参 数 new_link 改变 此 行为 并 使 mysql_connectO) 总 
是 打开 新 的 连接 ， 甚 至 mysql_connectO 曾 在 前 面 被 用 同样 的 参数 调用 过 
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( 续 表 ) 


可 选 。client flags 参数 可 以 是 以 下 常量 的 组 合 : 

MYSQL _CLIENT_SSI 一 使 用 SSL 加 密 

MYSQL _CLIENT_COMPRESS 一 使 用 压缩 协议 

MYSQL _CLIENT_IGNORE_SPACE 一 允许 函数 名 后 的 间隔 

MYSQL CLIENT INTERACTIVE 一 允许 关闭 连接 之 前 的 交互 超时 非 活动 时 间 


返回 值 : 如 果 成 功 ， 则 返回 一 个 MySQL 连接 标识 ， 失 败 则 返回 FALSE。 
WW mn 


脚本 一 结束 ， 到 服务 器 的 连接 就 被 关闭 ， 除 非 之 前 已 经 明确 调用 mysql_! 
close() 关 闭 了 。 


下 面 示例 演示 了 在 PHP 脚本 中 进行 MySQL 数据 库 服务 器 的 连接 ， 代 码 如 下 。 


<?php 

$con = mysql_connect("localhost","root",""): 

if (!$con) 

0 
//die0 函 数 输出 一 条 消息 ， 并 退出 当前 脚本 。 它 是 exit0 函 数 的 别名 
die(' 不 能 连接 ，' . mysql_error0): 。 //mysql_error0 函 数 返回 上 一 个 MySQL 函数 的 错 

误 文 本 ， 如 果 没 有 出 错 则 返回 "( 空 字符 串 ) 

} 

/ 关闭 数据 库 连接 .… 

mysql close($con); 

?> 


上 面 的 代码 使 用 了 mysql_connectO 函 数 连接 本 地 MySQL 数据 库 服务 器 ， 连 接 用 
户 名 为 root， 密 码 为 空 。 

在 实际 应 用 中 ， 通 常 在 多 个 脚本 文件 中 都 需要 进行 数据 库 的 连接 ， 此 时 为 了 维护 
方便 和 节省 代码 ， 可 将 数据 库 连 接 放 在 一 个 单独 的 文件 中 ， 在 需要 使 用 数据 库 连 接 的 
脚本 中 使 用 includeO 函 数 或 requireO 函 数 引用 该 文件 。 


9.1.2 断 开 与 MySQL 数据 库 的 连接 


通常 在 完成 数据 库 的 使 用 后 ， 需 要 断 开 与 MySQL 数据 库 服务 器 的 连接 。 通 常 使 
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用 mysql_closeO 函 数 来 断 开 与 MySQL 数据 库 服务 器 的 连接 ， 语 法 如 下 。 
Imysql close(link identifier) 
其 中 ， 各 参数 说 明 如 表 9-2 所 示 。 


必需 。MySQL 的 连接 标识 符 。 如 果 没 有 指定 ， 默 认 使 用 最 后 被 
mysql_connect() 打 开 的 连接 。 如 果 没 有 找到 该 连接 ， 函 数 会 尝 
调用 mysql_connectO 建 立 连接 并 使 用 它 。 如 果 发 生意 外 ， 没 有 找到 
连接 或 无 法 建立 连接 ， 系 统 发 出 E_WARNING 级 别 的 警告 信息 


返回 值 : 如 果 成 功 则 返回 tmue， 失 败 则 返回 false。 


WW Me 有 
1 通常 不 需要 使 用 mysql_close(), 因为 已 打开 的 非 持 久 连 接 会 在 脚本 执行 完 ! 
毕 后 自动 关闭 。mysql_close() 不 会 关闭 由 mysql_pconnect() 建 立 的 持久 连接 。 |! 


下 面 演示 了 在 PHP 脚本 中 关闭 一 个 由 mysql_connect() 函 数 所 建立 的 数据 库 连 接 。 
代码 如 下 。 
<?php 
$con = mysql connect("localhost","mysql_user","mysql pwd"); 
if(!$con) 


die(' 连 接 失 败 ， 原 因 : '. mysql_errorO); 
} 


/ 一 些 代码 … 


mysql_close($con); 
2 
上 面 的 代码 使 用 了 mysql_closeO 函 数 关 闭 一 个 已 创建 的 非 持久 数据 连接 。 
虽然 在 前 面 已 讲 过 ， 创 建 数据 库 连 接 的 脚本 一 结束 ， 其 数据 库 连 接 自动 关闭 。 但 
是 从 节省 服务 器 资源 层面 上 讲 ， 在 使 用 完 数据 库 连接 后 ， 使 用 mysql_close() 函 数 关闭 
数据 库 连 接 能 更 有 效 地 节省 服务 器 资源 。 


9.1.3 ”选择 和 使 用 MySQL 数据 库 


在 进行 数据 库 的 连接 后 ,需要 在 PHP 脚本 中 选择 需要 进行 操作 的 MySQL 数据 库 。 
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可 使 用 mysql_select_db() 函 数 ， 语 法 如 下 。 
mysql_select_db(database,connection) 


mySQL select db(0) 函 数 的 定义 和 用 法 如 下 。 

e@ Imysql select db() 函 数 设置 活动 的 MySQL 数据 库 。 

返回 值 : 如 果 成 功 ， 则 该 函数 返回 tue。 如 果 失 败 ， 则 返回 false。 
其 中 的 各 参数 说 明 如 表 9-3 所 示 。 


database 必需 。 规 定 要 选择 的 数据 库 
connection 可 选 。 规 定 MySQL 连接 。 如 果 未 指定 ， 则 使 用 上 一 个 连接 


下 面 示例 演示 了 在 PHP 脚本 中 选择 MySQL 数据 库 服 务 器 上 的 数据 库 , 代码 如 下 。 


<?php 
S$con = mysql_connect("localhost", "root", ""); 
if(!$con) 
{ 

die( 数 据 库 连接 失败 : '. mysql_errorO); 
} 


$db_selected = mysql_select_db("bookDb", $con);// 返 回 值 为 true 或 false 


if (1$db_selected) 
die ("选择 的 数据 库 不 存在 或 不 可 用 :" . mysql_errorO); 
} 


mysql_close($con); 
Ge 
在 上 面 的 代码 中 ， 使 用 当前 数据 库 连 接 选 择 bookDb 数据 库 作为 活动 数据 库 ， 对 
数据 库 的 所 有 操作 都 将 作用 于 该 活动 数据 库 。 大 家 可 以 看 到 ， 这 时 选择 数据 库 作 为 当 
前 活动 数据 库 ， 其 实 就 相当 于 在 MySQL 中 执行 use 命令 。 


9.1.4 执行 MySQL 指令 


进行 数据 库 的 操作 ， 在 PHP 中 需要 使 用 一 个 函数 来 执行 MySQL 指令 ， 这 就 是 
mysql query0 函 数 ， 语 法 如 下 。 


mysql query(query,connection) 
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mysql _ query() 函 数 的 定义 和 用 法 如 下 。 
e@ ”mysql query() 函 数 执行 一 条 MySQL 查询 。 
其 中 的 各 参数 说 明 如 表 9-4 所 示 。 


表 9-4 


必需 。 规 定 要 发 送 的 SQL 查询 。 注 释 : 查询 字符 串 不 应 以 分 号 结束 
connection 可 选 。 规 定 SQL 连接 标识 符 。 如 果 未 规定 ， 则 使 用 上 一 个 打开 的 连接 


说 明 DD Dl i ~ 
i 如 果 没 有 打开 的 连接 ， 本 函数 会 尝试 无 参数 调用 mysql_connect() 函 数 来 建 ; 
! 立 一 个 连接 并 使 用 。 ' 


返回 值 ，mysql_query0 仅 对 SELECT、SHOW、EXPLAIN 或 DESCRIBE 语句 返 
回 一 个 资源 标识 符 ， 如 果 查 询 执 行 不 正确 则 返回 false。 对 于 其 他 类 型 的 SQL 语句 ， 
mysql_query0) 在 执行 成 功 时 返回 tue， 出 错时 返回 false。 

非 false 的 返回 值 意味 着 查询 是 合法 的 并 能 够 被 服务 器 执行 。 这 并 不 能 说 明 任何 有 
关 影 响 到 的 或 返回 的 行 数 。 很 有 可 能 一 条 查询 执行 成 功 了 但 并 未 影响 到 或 并 未 返回 


该 函数 自动 对 记录 集 进行 读 取 和 缓存 。 如 需 运 行 非 缓存 查 询 ， 请 使 用 } 
mysql_unbuffered query(). | 


示例 1: 通过 mysql_query0 函 数 创建 一 个 新 数据 库 。 


<?php 
$con = mysql_connect("localhost","mysql_user","mysql pwd"); 
if(!$con) 
革 
die(' 无 法 连接 : .mysql_errorO); 
} 
$sql = "CREATE DATABASE my_db"; 
if (mysql query($sql,$con)) 
四 
echo "数据 库 my_db 创建 成 功 !"; 


echo "数据 库 创建 过 程 中 发 生 问题 : " . mysql_error0; 
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Bb 


ye 


除了 mysql_query0 函 数 能 够 执行 SQL 语句 外 ，PHP 还 提供 了 另 一 个 函数 
mysql db query(0 。 该 函数 与 mysql query0 函数 具有 相同 的 功能 ， 其 区 别 在 于 
mysql_db_query() 函 数 在 执行 SQL 语句 时 可 以 同时 选择 数据 库 。 
示例 2: 执行 查询 语句 。 
<?php $con= mysql connect("localhost","mysql user","mysql pwd"); 
if(!$con) { 
die(' 无 法 连接 :'. mysql_errorO); 
} 
$sql = "SELECT * FROM Person"; 
mysql query($sql,$con); 
1/ 一 些 代码 
mysql close($con); 
?> 


9.1.5 ”操作 结果 集 


1. 获取 影响 的 行 数 
在 每 一 次 成 功 执行 SQL 查询 后 ， mysql_query0O 函 数 总 是 会 返回 一 个 结果 集 。 要 对 
结果 集 进行 处 理 ， 首 先 需要 获取 所 执行 的 SQL 语句 影响 的 行 数 。 
对 于 结果 集中 包含 的 记录 数 , 可 使 用 mysql_num _rows() 函 数 获取 。 语法 格式 如 下 。 
mysql num rows(data) 
其 中 的 各 参数 及 说 明 如 表 9-5 所 示 。 
表 9-5 
数 描 述 
必需 。 结 果 集 。 该 结果 集 从 mysql_query0 的 调用 中 得 到 


mysql num rows() 返 回 结果 集中 行 的 数目 。 此 命令 仅 对 SELECT 语句 有 效 。! 
要 取得 被 INSERT、UPDATE 或 者 DELETE 查询 所 影响 到 的 行 的 数目 ， 用 ， 
mysql affected rows(). 1 


其 中 ， 参 数 data 为 函数 mysql query0 所 返回 的 结果 集 。 函 数 返 回 结果 集中 的 
记录 数 。 
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下 面 演示 了 在 PHP 脚本 中 获取 结果 集中 的 记录 数 ， 代 码 如 下 。 


<2php 

@mysql_connect("localhost","root","") or die(" 数 据 库 连接 失败 "); 
@mysql_select_db("bookdb") or die(" 选 择 的 数据 库 不 存在 或 不 可 用 "); 
mysql_query("set names gb2312"); 

Smyquery = mysql_query("select * from bookInfo"); 

Srowscnt = mysql num rows($myquery): 

echo "结果 集中 的 行 数 为 : ".$Srowscnt; 

mysql_close(); 

?> 


如 果 在 mysql_query0 函 数 中 使 用 INSERT、UPDATE 和 DELETE 语句 ， 应 使 用 
mysql_affected_rows() 函 数 获取 所 影响 到 的 记录 数 ， 其 语法 如 下 。 


mysql_affected_rows(link identifier) 


其 中 的 各 参数 说 明 如 表 9-6 所 示 。 
表 9-6 


必需 。MySQL 的 连接 标识 符 。 如 果 没 有 指定 , 默认 使 用 最 后 被 mysql_connect0 
打开 的 连接 。 如 果 没 有 找到 该 连接 ， 函 数 会 尝试 调用 mysql_connectO 建 立 连接 
并 使 用 它 。 如 果 发 生意 外 ， 没 有 找到 连接 或 无 法 建立 连接 ， 则 系统 会 发 出 
E_WARNING 级 别 的 警告 信息 


其 中 ， 参 数 link identifier 为 已 打开 的 数据 库 连 接 标识 符 。 如 果 未 设置 该 参数 ， 函 
数 默 认 使 用 上 一 次 所 打开 的 数据 库 连 接 ， 如 果 未 找到 连接 ， 函 数 将 尝试 以 无 参数 方式 
调用 mysql_connect() 函 数 建立 数据 库 连 接 并 使 用 ， 如 果 发 生意 外 ， 如 找 不 到 数据 库 连 
接 或 创建 数据 库 连接 失败 时 ， 将 产生 一 条 警告 信息 。 

函数 返回 由 参数 link_identifier 所 关联 的 数据 库 连 接 进 行 的 INSERT、UPDATE 和 
DELETE 语句 所 影响 到 的 行 数 。 函 数 执行 成 功 则 返回 最 近 一 次 操作 所 影响 的 行 数 ， 若 最 
近 一 次 查询 失败 ， 则 返回 -1。 在 使 用 UPDATE 语句 时 ，MySQL 不 会 将 原 值 与 新 值 一 样 
的 列 进行 更 新 ,因此 该 函数 所 返回 的 值 不 一 定 就 是 使 用 mysql_query0 函 数 所 影响 到 的 行 
数 ， 此 时 只 是 返回 真正 被 更 新 的 行 数 。 


2. 获取 结果 集中 的 数据 


要 显示 使 用 mysql_query0 函 数 所 返回 的 结果 集 ， 首 先 需 获取 结果 集中 的 数据 ， 获 
取 结 果 集 中 的 某 一 条 数据 使 用 mysql_result0 函 数 ， 语 法 如 下 。 


mysql_result(data,row ,field) 


其 中 的 各 参数 及 说 明 如 表 9-7 所 示 。 


link_identifier 
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必需 。 规 定 要 使 用 的 结果 集 。 该 标识 符 是 mysql_query0 函 数 返 回 的 结 
果 集 

必需 。 规 定 行 号 。 行 号 从 0 开始 

可 选 。 规 定 获取 哪个 字段 ,可 以 是 字段 偏 移 值 、 字 段 名 或 table.fieldname 
如 果 该 参数 未 规定 ， 则 该 函数 从 指定 的 行 获取 第 一 个 字段 


下 面 示例 演示 了 在 PHP 脚本 中 显示 某 一 条 记录 ， 代 码 如 下 。 
先 在 MySQL 中 创建 bookInfo 表 ， 并 插入 测试 数据 如 图 9-1 所 示 。 


Dlboora [bookNane [bookAuthor 
[a | 101|PHP 怎 么 学 “| 毛利 38.5 
Lm | 102| 我 的 大 学 |coco 54.5 


| (Auto) | (NULL) | (NULL) (NULL) 
9-1 


后 面 的 代码 将 以 此 表 为 基础 进行 增 、 删 、 改 、 查 操作 。 


<?php 

@mysql_connect("localhost","root","123456") or die(" 数 据 库 连接 失败 ! "); 
@mysql_select_db("bookdb") or die(" 选 择 的 数据 库 不 存在 或 不 可 用 !"); 
mysql query("set names gb2312"); 

$myquery = @mysql_query("select * from bookInfo") or die("SQL 语句 执行 失败 1"); 
/下 面 使 用 mysql_result0 函 数 获取 结果 集中 第 一 行 的 数据 信息 

echo "图 书 编号 :" . mysql_result(Smyquery, 0, 0) . 

echo "图 书 名 称 :" . mysql_result($myquery, 0. 1) . 

echo "图 书 作 者 :" . mysql_result(Smyquery, 0, 2) . 

echo "图 书 价格 :" . mysql_result($myquery, 0. 3) . " 

mysql_close(); 

re 


ER 9-2 所 示 。 


上 上面 的 代码 实现 了 获取 结果 集中 的 某 一 行 数据 ， 结 合 使 用 mysql_num rows 函数 
所 返回 的 行 数 ， 可 以 输出 结果 集中 的 所 有 数据 。 
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下 面 示例 演示 了 返回 结果 集中 的 所 有 信息 ， 代 码 如 下 。 


<?php 

@mysql_connect("localhost","root","123456") or die(" 数 据 库 连 接 失 败 ! "; 

@mysql_select_db("bookdb") or die(" 选 择 的 数据 库 不 存在 或 不 可 用 !"); 

mysql_query("set names gb2312"); 

Smyquery = @mysql_query("select * from bookInfo") or die("SQL 语句 执行 失败 1"); 

Srowscnt = mysql_ num rows($myquery): 

echo "<table border=\"1\"><tr><th> 图 书 编号 </th><th> 图 书 名 称 </th><th> 图 书 作者 
</th><th> 图 书 价格 </th></tr>"; 

for($i=0; $i < $rowscnt; $i++){ 

echo "<tr><td>" . mysql_result($myquery, $1, 0) . "</td>"; 

echo "<td>" . mysql result($myquery, $i, 1) . "</td>"; 

echo "<td>" . mysql_result($myquery, $1, 2) . "</td>"; 

echo "<td>" . mysql_result($myquery, $1, 3) . "</td>"; 

hs 

echo "</table>"; 

mysql_close(); 


图 书 名 称 
PHP 怎 么 学 
我 的 大 学 
酷 粮 的 医生 
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9.2 操作 MySQL 数据 库 中 的 数据 


在 Web 中 ,常常 需要 用 户 在 浏览 器 上 通过 表单 对 数据 库 中 的 数据 进行 操作 ， 如 添 
加 数据 记录 、 更 新 数据 记录 和 删除 数据 记录 等 。 下面 将 对 用 户 在 HTML 表单 上 对 数据 
进行 操作 ， 然 后 提交 到 服务 器 并 使 用 mysql_queryO 函 数 执行 SQL 语句 的 方式 操作 数 
据 进 行 详细 讲解 。 


9.2.1 添加 数据 


服务 器 在 接收 到 用 户 的 数据 后 ， 采 用 mysql_query() 函 数 执行 相应 的 insert 语句 ， 
将 用 户 输入 的 数据 添加 到 数据 库 。 
下 面 将 演示 一 个 完整 的 添加 数据 过 程 ，HIML 页 面 代码 如 下 。 
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<!DOCTYPE html PUBLIC "-W/W3CWDTD XHTML 1.0 Transitional//EN" "http:/www. 
w3.org/TR/xhtmll/DTD/xhtml1-transitional.dtd"> 

<html xmlns="http://www-w3.org/1999/xhtml"> 

<head> 

<meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> 

<title> 添 加 数据 </title> 

</head> 


<body> 
<!-- 这 里 指示 将 表单 提交 到 addBook.php 页 面 中 --> 
<form id="forml" name="forml" method="post" action="addBook.php"> 
<table width="512" border="1" align="center"> 
<caption><font size="5"><b> 添 加 图 书信 息 </b></font></caption> 
<tr> 
<td width="112"> 图 书 编号 : </td> 
<td width="400"><input name="bookId" type="text" id="bookId" /></td> 
</tr> 
<tr> 
<td> 图 书 名 称 : </td> 
<td><input name="bookName" type="text" id="bookName" /></td> 
</tr> 
<tr> 
<td> 图 书 作者 : </td> 
<td><input type="text" name="bookAuthor" id="bookAuthor" /></td> 
</tr> 
<tr> 
<td> 图 书 价格 : </td> 
<td><input name="bookPrice" type="text" id="bookPrice"/></td> 
</tr> 
<tr> 
<td>&nbsp;</td> 
<td><input type="submit" name="submit" id="submit" value=" 提 交 " /></td> 
</tr> 
</table> 
</form> 
</body> 
</html> 


在 页 面 上 添加 数据 ， 单 击 “ 提 交 ” 按 钮 后 跳 转 到 “addBook.php” 页 面 进行 处 理 ， 
如 图 9-4 所 示 。 


使 用 PHP 开发 Web 应 用 程序 


添加 图 书信 息 


PHP 页 面 代码 如 下 。 


<?php 

S$bookId = $_POST["bookId']; 

SbookName =$_POST[bookName']; 

S$bookAuthor =$ POST[bookAuthor]; 

S$bookPrice = $_ POST["bookPrice']; 

Sins_sql] = "insert into bookInfo values('$bookId', '$bookName', '$bookAuthor', '$bookPrice’)"; 
@mysql_connect("localhost","root","") or die(" 数 据 库 连接 失败 ! "); 
@mysql_select_db("bookDb") or die(" 选 择 的 数据 库 不 存在 或 不 可 用 !"); 
mysql_query("set names gb2312"); 

Smyquery = mysql_query($ins_sq]); 

if($myquery){ 

echo "插入 数据 成 功 !"; 

Jelse{ 

echo "插入 数据 失败 ! "; 

yb 

mysql_close(); 

?> 


添加 成 功 界面 ， 如 图 9-5 所 示 。 


9.2.2 ”修改 数据 


先 创建 一 个 查询 的 PHP 页 面 browseBookInfo php， 代 码 如 下 。 
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<script language="javascript"> 
function chk(id){ 
if(confirm(" 确 定 要 删除 该 资料 ? ")){ 
window.location="deleteBookAction.php?id="+id; 
jelse{ 
return false; 
} 
} 
</script> 
<?php 
@mysql_connect("localhost","root",") or die(" 数 据 库 连接 失败 ! "); 
@mysql_select_db("bookdb") or die(" 选 择 的 数据 库 不 存在 或 不 可 用 !"); 
mysql_query("set names gb2312"); 
Smyquery = @mysql_query("select * from bookinfo") or die("SQL 语句 执行 失败 1"); 
Spage_ size = 3; 
$num cnt = mysql num rows($myquery); 
S$page_cnt = ceil($num cnt / $page size); 


iflisset($_GET['p]){ 
Spage = $_GET['p"]; 
}else{ 
Spage =1; 
} 
$query_start = ($page - 1) * Spage_size; /计算 每 页 开始 记录 号 
$querysql = "select * from bookInfo limit $query_start, Spage_size":// 组 成 SQL 语句 
$queryset = mysql_ query($querysqD: /执行 SQL 语句 
echo "<table align='center border=\"1\"><tr><th> 图 书 编号 </th><th> 图 书 名 称 </th>< 了 b> 图 书 
作者 </th><th> 图 书 价格 </th><th> 操 作 </th></tr>"; 
while($row = mysql_fetch_array($queryset MYSQL BOTH)){ V// 逐 行 从 数据 集 获取 数据 
echo "<tr><td>" . $row[0] . "</td>"; 
echo "<td>" . $row[1] . "</td>"; 
echo "<td>" . Srow[2] . "</td>"; 
echo "<td>" . Srow[3] . "</td>"; 
echo "<td><a href="updateBookInfo.php?id=$row[0] 人 > 修改 </a> 
<a href='#' onclick='chk($row[0]); 人 > 删除 </a> 
</td></tr>"; 
} 
echo "</table><br>"; 
Spager = " 共 S$page_cnt 页 跳 转 至 第 "; 
if($page_cnt > D){ 
for($i=1; $i <= $page_cnt; Si++){ 
if($page — $D{ 
Spager .= "<a href='?p=$i'><b>$i</b></a> "; 
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}else{ 
$pager .= "<a href—"?p=$i'>$1</a> "; 
} 
echo $pager . ”页 "; 
mysql_close(); 
ye 


效果 如 图 9-6 所 示 。 


D booknfo 
€ 3 © ©® 12700est/browseBookinio.php 
灌 负 上 尖 5 下 国 要 % 算 澡 琴 度 -下 治 百度 


图 书 编号 。 图 书 名 称 。 “图 书 作者 图书 价 格 
101 PHP 怎 么 学 毛利 38.5 
102 我 的 大 学 coco 545 


103 本 炫 的 医生 。。” 程 康 553 
共 2 页 跳 转 至 第 2 页 


图 9-6 


单 击 “ 修 改 ” 按 钮 ， 修 改 数据 代码 如 下 。 


<?php 
iflisset($_GET['id])){ 
$id = $_GET['id']; 
@mysql_connect("localhost","root","") or die(" 数 据 库 连接 失败 ! "); 
@mysql_select_db("bookdb") ”or die(" 选 择 的 数据 库 不 存在 或 不 可 用 !"); 
mysql_query("set names gb2312"); 
$sql = "select * 位 om bookinfo where bookid ="$id"™"; 
Smyquery = @mysql_query($sql) or die("SQL 语句 执行 失败 1"); 
Srow = mysql_fetch_array($myquery, MYSQL _ BOTH); 
echo <<<Eof 
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http:/Wwww.w3.org/TR/xhtmll/DTD/xhtmll-transitional.dtd"> 
<html xmlns="http://www.w3.org/1999/xhtml"> 
<head> 
<meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> 
<title> 修 改 数 据 </title> 
</head> 
<form action="updateBookAction.php" method="post" name="updinfo"> 
<table width="400" border="1"> 
<tr> 
<td> 图 书 编号 : </td> 
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<td>$row[0]<input name="bookId" type="hidden" value="$row[0]" /></td> 
</tr> 
<tr> 

<td> 图 书 名 称 : </td> 

<td><input name="bookName" type="text" value="$row[1]" /></td> 
</tr> 
<tr> 

<td> 图 书 作 者 : </td> 

<td><input name="bookAuthor" type="text" value="$row[2]" /></td> 
</tr> 
<tr> 

<td> 图 书 价格 : </td> 

<td><input name="bookPrice" type="text" value="$row[3]"/></td> 


<td><input name="submit" type="submit" value=" 提 交 " /></td> 
</tr> 
</table> 


Eof; 

mysql_close(); 

jelse{f 

echo "ID 号 错误 ， 请 <a href-'browseBookImnfo.php> 浏 览 </a>": 
} 

?> 


页 面 效果 如 图 9-7 所 示 。 


人 i a 
DD 127.0.0.1:8020/seven_test/twc x 


€ > C ©® 12700.18020/test/updateBookinfo.php?id=102 
让 应 用 党 百度 一 下 屁 刘 算 ”党 百 度 - 下 党 吾 度 


102 
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当 用 户 单 击 “ 提 交 ” 按 钮 后 ， 将 表单 提交 到 “updateBookAction.php” 页 面 ， 该 页 
面 代码 如 下 。 


<?php 

S$bookId=$_ POST['id]; 

SbookName = $_POST['bookName']; 

S$bookAuthor = $_ POST[bookAuthor]: 

$bookPrice = $_ POST["bookPrice']; 

S$upd_sql = "update bookinfo set bookName = '$bookName', book Author = '$bookAuthor', 
bookPrice = '$bookPrice' where bookId = '$bookId'": 
@mysql_connect("localhost","root","") or die(" 数 据 库 连接 失败 ! "); 
@mysql_select_db("bookdb") or die(" 选 择 的 数据 库 不 存在 或 不 可 用 !"); 
mysql_query("set names gb2312"); 

$myquery = mysql_query($upd sq)); 

if($myquery){ 

echo "更 新 数据 成 功 !"; 

jelsef 

echo "更 新 数据 失败 !"; 

} 

echo "<a href='browseBookInfo.php'> 浏览 </a>"; 

mysql close(); 

?> 


修改 成 功 后 ， 效 果 如 图 9-8 所 示 。 


/testl/updateBookAction php 


9.2.3 ”删除 数据 


运行 browseBookInfo .php 文件 ， 当 用 户 单 击 数据 行 的 “删除 ” 超 链接 后 ， 弹 出 如 
图 9-9 所 示 的 对 话 框 。 
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€ > CC @17700l/iestUbrowsesookinfophp 
汪汪 Ee 


127001:8020 号 示 
图“ 了 SR 


DZ | SKY 人 coco | 
103 。 | 酷 过 的 医生 
共 2 页 跳 转 至 获 1 2 页 
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当 单 击 “ 确 定 ” 按 钮 后 ， 会 调用 javascript 中 的 代码 ， 调 用 deleteBookAction.php 
文件 ， 进 行 删除 操作 ， 该 文件 代码 如 下 。 


<?php 

$bookId =$_GET[bookId']: 

$upd_sql = "delete from bookInfo where bookId='$bookId'"; 
@mysql_connect("localhost","root","") or die(" 数 据 库 连接 失败 ! "); 
@mysql_select_db("bookdb") or die(" 选 择 的 数据 库 不 存在 或 不 可 用 !"); 
mysql_query("set names gb2312"); 

Smyquery = mysql query(Supd_ sqD; 

if($myquery){ 

echo "删除 数据 成 功 !"; 

Jelse{ 

echo "删除 数据 失败 !"; 

} 

echo "<a href='browseBookInfo.php'> 浏览 </a>"; 

mysql_close(); 

?> 


删除 成 功 后 结果 如 图 9-10 所 示 。 


机 除数 据 成 功 | .浏览 


9.2.4 获取 数据 库 的 信息 


可 以 使 用 mysql_list dbs() 函 数 获取 MySQL 服务 器 的 数据 库 列表 信息 , 语法 如 下 。 
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mysql list_dbs(connection) 


其 中 的 各 参数 说 明 如 表 9-8 所 示 。 


connection 可 选 。 规 定 SQL 连接 标识 符 。 如 果 未 规定 ， 则 使 用 上 一 个 打开 的 连接 


mysql list_dbs() 将 返回 一 个 结果 指针 , 包含 了 当前 MySQL 进程 中 所 有 可 用 的 
数据 库 。 ' 
用 mysql_tablename() 函 数 来 遍历 此 结果 指针 ， 或 者 任何 使 用 结果 表 的 函数 ， ; 
例如 mysql_fetch array()。 ' 


下 面 示例 演示 了 列 出 MySQL 服务 器 上 的 所 有 可 用 数据 库 ， 代 码 如 下 。 


<?php 

@mysql_connect("localhost","root","") or die(" 数 据 库 连接 失败 ! "); 
Sdbs = mysql_list_ dbsO; 

echo "<h3>MySQL 服务 器 的 数据 库 信 息 如 下 : </h3>"; 
while($dbrow = mysql fetch array($dbs, MYSQL BOTHD)){ 

echo $dbrow[0]."<br>"; 

} 

mysql_close(); 

yy 上 


运行 结果 如 图 9-11 所 示 。 


.0.0. 1/testl/index. php 
MYSQL 服务 器 的 数据 库 信息 如 下 : 


information_schema 
bookdb 

msql 

test 


你 可 能 会 发 现 , 使 用 这 种 方式 列 出 服务 器 数据 库 列 表 与 你 在 MySQL 命令 行使 用 “show 
databases; ”命令 的 结果 是 一 样 的 。 
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【单元 小 结 】 


e 了 解 PHP 连接 MySQL 数据 库 的 方式 
e PHP 对 MySQL 中 的 数据 进行 操作 


【单元 自 测 】 


1. 内 关联 (inner join) 是 用 来 做 什么 的 ? ( ” ) 

A. 把 两 个 表 通 过 相同 字段 关联 入 一 张 持 久 的 表 中 

B. 创建 基于 两 个 表 中 相同 行 的 结果 集 

C. 创建 基于 一 个 表 中 的 记录 的 数据 集 

D. 创建 一 个 包含 两 个 表 中 相同 记录 和 一 个 表 中 全 部 记录 的 结果 集 
. 以 下 哪个 DBMS 没有 PHP 扩展 库 ?(  ) 

A. MySQL 

B. IBM DB/2 

C. Microsoft SQL Server 

D. 以 上 都 不 对 
3. 考虑 如 下 脚本 。 假 设 mysql_query 函数 将 一 个 未 过 滤 的 查询 语句 送 入 一 个 已 经 

打开 的 数据 库 连 接 ， 以 下 哪个 选项 是 对 的 ? ( 双 选 ) ( ) 


<?php 
Sr = mysql_query (DELETE FROM MYTABLE WHERE ID='.$_GET[TD']): 


A. MYTABLE 表 中 的 记录 超过 1 条 

B. 用 户 输入 的 数据 需要 经 过 适当 的 转 义 和 过 滤 

C. 调用 该 函数 将 产生 一 个 包含 了 其 他 记录 条 数 的 记录 

D. 给 URL 传递 ID=0+OR+1 将 导致 MYTABLE 中 的 所 有 表 被 删除 
. 以 下 哪个 说 法 正确 ? (  ) 

A. 使 用 索引 能 加 快 插入 数据 的 速度 

B. 良好 的 索引 策略 有 助 于 防止 跨 站 攻击 

C. 应 当 根 据 数据 库 的 实际 应 用 原理 设计 索引 

D. 删除 一 条 记录 将 导致 整个 表 的 索引 被 破坏 
5. join 能 否 被 嵌 套 ? ( ” ) 

A. 能 

B. 不 能 
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【上 机 实战 】 
上 机 目标 
e 了 解 PHP 连接 MySQL 数据 库 的 方式 
e PHP 对 MySQL 中 的 数据 进行 操作 


上 机 练习 


多 第 一 阶段 多 


练习 : PHP 连接 和 断 开 MySQL 数据 库 ， 查 询 结果 集 。 


【问题 描述 】 


现 有 用 户 表 userInfo， 需 要 对 表 中 的 数据 进行 增 、 删 、 改 、 查 操作 ， 下 面 示例 描 
述 了 连接 数据 库 ， 并 对 数据 库 中 的 userInfo 表 中 的 数据 进行 操作 。 
代码 如 下 。 


<?php 

@mysql_connect("localhost","root","") 

or die(" 数 据 库 连接 失败 ! "); 

@mysql_select_db("userdb") 

or die(" 选 择 的 数据 库 不 存在 或 不 可 用 !"); 

mysql_query("set names gb2312"); 

$myquery = @mysql query("select * 位 om userinfo") 

or die("SQL 语句 执行 失败 1"); 

Srowscnt = mysql_num rows($myquery): 

echo "<table border=\"1\"><tr><th>id</th><th> 姓 名 </th><th> 性 别 </ 了 ><th> 地 址 </th><th> 
邮件 </th></tr>"; 

for($i=0; $i < $rowscnt; $i++){ 

echo "<tr><td>" . mysql result($myquery. $i, 0) . "</td>"; 

echo "<td>" . mysql result($myquery, $i, 1) . "</td>"; 

echo "<td>" . mysql result($myquery, $i, 2) . "</td>"; 

echo "<td>" . mysql_result($myquery, $i, 3) . "</td>"; 

echo "<td>" . mysql result($myquery, $i, 4) . "</td></tr>"; 
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echo "</table>"; 
mysql_close(); 
> 


运行 后 效果 如 图 9-12 所 示 。 


只 名 性 别 | 
101 号 三 田 ” 洒 北 武汉 
102 订 四 区 ”网 南 长 沙 
os 号 明 [ 男 | 


9-12 


令 第 二 阶段 多 


练习 : 修改 “第 一 阶段 练习 ”中 的 代码 ， 增 加 “修改 5“ 删除 ”“ 添 加 ” 


功能 。 


【问题 描述 】 


完善 上 例 的 功能 ， 增 加 “添加 "” “修改 "” “删除 ”功能 。 
修改 和 删除 功能 如 图 9-13 所 示 。 


| 电子 邮件 | 操作 
Thansshantsina con 峰 广 下 村 


[ahiecahu com 


参考 代码 如 下 。 


<script language="javascript"> 

function chk(id){ 

if(confirm(" 确 定 要 删除 该 资料 ?")){ 
window.location="deleteUserInfoAction.php?id="+id; 
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} 
} 
</script> 
<?php 
@mysql_connect("localhost","root","") or die(" 数 据 库 连接 失败 ! "); 
@mysql_select_db("userdb") or die(" 选 择 的 数据 库 不 存在 或 不 可 用 !"); 
mysql_query("set names gb2312"); 
$myquery = @mysql_query("select * ffom userinfo") or die("SQL 语句 执行 失败 1"); 
Spage_size = 3; 
Snum cnt = mysql num rows($myquery); 
$page_cnt = ceil($num cnt / $page size); 


iflisset($_GET['p]){ 
Spage = $_GETI['p']; 
}else{ 
Spage = 1; 
$query_start = ($page - 1) * $page_size; /计算 每 页 开始 记录 号 
S$querysql = "select * from userInfo limit $query_start, $page_size":// 组 成 SQL 语句 
$queryset = mysql_query($querysqD; /执行 SQL 语句 
echo "<table align='center border=\"1\"><tr><th>ID</th><th> 姓 名 </th><th> 性 别 </th><th> 
地 址 </th><th> 电 子 邮 件 </th><th> 操 作 </th></tr>"; 
while($row = mysql_fetch_array($queryset, MYSQL _ BOTH)){ V/ 逐 行 从 数据 集 获 取 数据 
echo "<tr><td>" . $row[0] . "</td>"; 
echo "<td>" . Srow[1] . "</td>"; 
echo "<td>" . Srow[2] . "</td>"; 
echo "<td>" . Srow[3] . "</td>"; 
echo "<td>" . $row[4] . "</td>"; 
echo "<td><a href='updateUserInfo.php?id=$row[0]> 修 改 </a> 
<a hre 人 ='# onclick='chk($Srow[0]):> 删 除 </a> 
</td></tr>"; 
} 
echo "</table><br>"; 
Spager = " 共 $page_cnt 页 跳 转 至 第 ": 
if($page_cnt > D){ 
for($i=1; $1 <= $page_cnt; $i++){ 
if($page — 3D){ 
Spager = "<a href="?7p=$i'><b>$i</b></a> ": 
Jelse{ 
Spager .= "<a hre 人 ='?p=$1>$i</a> "; 
3} 
echo Spager . ”页 "; 
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D 
mysql_closeO; 
ye 


其 他 的 代码 大 家 参考 理论 部 分 知识 点 。 


【拓展 作业 】 


1. 使 用 PHP 连接 和 断 开 MySQL 数据 库 。 
2. 新 建 一 个 student 表 ， 包 含 学 号 、 姓 名 、 性 别 、 年 龄 字段 ， 然 后 对 表 中 的 数据 
进行 增 、 删 、 改 、 查 操作 。 


Cookie、Session 及 


图 像 处 理 


里 课程 目标 


入 了 解 Cookie、Session 的 概念 
PP 掌握 Cookie、Session 的 操作 与 应 用 
P 掌握 PHP 中 的 图 像 处 理 方法 
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@ PEn 


在 Web 系统 中 ， 常 常 需要 记录 用 户 的 有 关 信息 ， 以 供用 户 再 次 以 此 身份 对 Web 
服务 器 提起 请 求 时 进行 确认 。 在 PHP 中， 通常 采用 的 方式 就 是 使 用 Cookie 或 Session 
来 保存 用 户 信息 。 

Web 系统 是 采用 HTTP 协议 进行 数据 传输 的 。 而 该 协议 是 一 个 无 状态 协议 ， 无 法 
得 知 用 户 的 浏览 状态 ， 也 就 是 说 ， 客 户 端 与 服务 器 的 每 一 次 连接 都 被 当成 是 一 次 单独 
的 操作 。 用 户 在 前 一 张 网 页 的 数据 不 能 在 第 二 张 网 页 上 使 有 用。 因此， 产生 了 两 种 用 于 
保持 连接 状态 的 技术 ， 它 们 就 是 Cookie 和 Session。 


10.1 概述 


10.1.1 ”Cookie 的 概念 


Cookie 是 一 种 在 远程 浏览 器 端 存储 数据 并 以 此 来 跟踪 和 识别 用 户 的 机 制 。 当 相同 
的 计算 机 通过 浏览 器 请 求 一 个 页 面 时 ， 原 先 存储 的 Cookie 也 会 发 送 到 服务 器 。 由 于 
Cookie 是 保存 在 客户 端的 ， 因 此 可 以 随意 地 设置 Cookie 的 保存 时 间 。 为 了 能 够 永久 
地 保存 用 户 信息 ， 采 用 Cookie 是 最 为 便捷 的 方式 。 

要 使 用 Cookie， 必 须知 道 其 工作 原理 。 一 般 来 说 ，Cookie 通过 HTTP Headers 从 
服务 器 端 返回 到 浏览 器 上 。 首 先 ， 服 务 器 端 在 响应 中 利用 set-Cookie header 来 创建 一 
个 Cookie。 然 后 浏览 器 在 它 的 请 求 中 通过 Cookie header 包含 这 个 已 经 创建 的 Cookie， 
并 且 将 它 返回 到 服务 器 ， 从 而 完成 浏览 器 的 验证 。 


在 实际 传递 过 程 中 ，Cookie 值 是 自动 进行 URL 编码 的 ; 当 收 到 Cookie 时 ，1 
自动 进行 URL 解码 。 如 果 不 希 望 在 发 送 Cookie 时 进行 URL 编码 ， 可 使 用 ， 
setrawcookie() 函 数 替 代 。 i 


如 果 用 户 浏览 器 不 支持 Cookie, 或 者 用 户 在 浏览 器 中 设置 禁止 Cookie， 则 Cookie 
将 不 能 建立 ， 此 时 需要 使 用 其 他 方式 将 一 个 页 面 的 信息 传递 至 另 一 个 页 面 ， 可 以 采用 
Form 表单 提交 数据 的 方式 。 

一 个 浏览 器 能 创建 的 Cookie 数量 最 多 为 30 个 ， 并 且 每 个 不 能 超过 4KB， 每 个 
Web 站 点 所 能 设置 的 Cookie 总 数 不 能 超过 20 个 。 因 浏览 器 多 种 多 样 ， 各 种 浏览 器 对 
Cookie 的 处 理 也 不 同 ， 因 此 在 使 用 Cookie 时 一 定 要 考虑 到 这 个 因素 。 
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10.1.2 ”Session 的 概念 


PHP Session 变量 用 于 存储 有 关 用 户 会 话 的 信息 , 或 更 改 用 户 会 话 的 设置 。Session 
变量 保存 的 信息 是 单一 用 户 的 ， 并 且 可 供应 用 程序 中 的 所 有 页 面 使 用 。 


1. PHP Session 变量 


当 用 户 运 行 一 个 应 用 程序 时 ， 会 打开 它 ， 做 些 更 改 ， 然 后 关闭 它 ， 这 很 像 一 次 会 
话 。 计 算 机 清楚 你 是 谁 。 它 知道 你 何 时 启动 应 用 程序 ， 并 在 何 时 终止 。 但 是 在 因特网 
上 ， 存 在 一 个 问题 一 一 服务 器 不 知道 你 是 谁 以 及 你 做 什么 ， 这 是 因为 HTTP 地 址 不 能 
维持 状态 。 

通过 在 服务 器 上 存储 用 户 信息 以 便 随后 使 用 , PHP Session 解决 了 这 个 问题 (如 用 户 名 
称 、 购 买 商品 等 )。 不 过 ， 会 话 信息 是 临时 的 ， 在 用 户 离 开 网 站 后 将 被 删除 。 如 果 需 要 永 
久 存 储 信息 ， 可 以 把 数据 存储 在 数据 库 中 。 

Session 的 工作 机 制 是 : 为 每 个 访问 者 创建 一 个 唯一 的 id(UID)， 并 基于 这 个 UID 
来 存储 变量 。UID 存储 在 cookie 中 ， 或 通过 URL 进行 传导 。 

由 于 Session 是 以 文本 文件 的 形式 存储 在 服务 器 端的 ,所 以 不 怕 客 户 端 修改 Session 
的 内 容 ， 实 际 上 存放 在 服务 器 端的 Session 文件 ，PHP 将 自动 修改 其 权限 ， 只 保留 了 
系统 读 和 写 的 权限 ， 并 且 不 能 通过 FTP 方式 进行 修改 ， 所 以 相对 比较 安全 。 


2. Session 实现 机 制 


在 Session 的 实现 中 采用 了 Cookie 技术 ，Session 会 在 客户 端 保存 一 个 包含 
session id(Session 编号 ) 的 Cookie; 在 服务 器 端 保存 其 他 Session 变量 , 如 session_ name 
等 。 当 用 户 请 求 服务 器 时 也 把 session_ id 一 起 发 送 到 服务 器 中 , 通过 session id 提取 所 
保存 在 服务 器 端的 变量 ， 就 能 识别 用 户 了 ， 同 时 也 不 难 理解 为 什么 Session 有 时 会 
失效 了 。 

如 果 用 户 浏览 器 设置 禁止 Cookie, UNIX/Linux 系列 的 主机 将 可 以 自动 检查 Cookie 
状态 , 并 且 自 动 将 session_id 加 在 URL 后 面 传递 至 服务 器 , 而 Windows 系统 的 主机 却 
没有 此 功能 。 

Cookie 机 制 采用 的 是 客户 端 保持 状态 的 方案 ， 而 Session 则 是 采用 服务 器 端 保持 
状态 的 方案 。 采 用 后 者 需要 在 客户 端 保存 一 个 标识 ， 所 以 Session 机 制 可 能 需要 借助 
于 Cookie 机 制 来 达到 保存 标识 的 目的 。 但 也 有 其 他 选择 ， 如 有 些 网 站 常用 的 一 种 叫 
URL 重 写 的 方式 ， 就 是 把 session id 直接 附加 在 URL 后 面 。 这 两 种 方式 各 有 千秋 ， 大 
家 可 以 在 实际 应 用 中 自行 选择 。 


10.2 Cookie 操作 与 应 用 


Cookie 可 用 于 保存 用 户 状态 。 在 PHP 中 ， 可 直接 对 Cookie 进行 操作 ， 如 将 状态 
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信息 写 入 Cookie、 从 Cookie 读 取 状 态 信息 、 设 置 用 户 状态 等 信息 的 保存 时 间 等 。 


10.2.1 设置 Cookie 


在 PHP 中 ,对 Cookie 的 操作 基本 都 是 通过 学 习 setcookie() 函 数 来 实现 的 ,setcookieO 
函数 向 客户 端 发 送 一 个 HTTP cookie， 其 语法 如 下 。 


setcookie(name,value,expire,path,domain,secure) 
其 中 的 各 参数 及 说 明 如 表 10-1 所 示 。 


表 10-1 
描 述 

必需 。 规 定 cookie 的 名 称 
必需 。 规 定 cookie 的 值 
可 选 。 规 定 cookie 的 有 效 期 
可 选 。 规 定 cookie 的 服务 器 路 径 
可 选 。 规 定 cookie 的 域名 
可 选 。 规 定 是 否 通过 安全 的 HTTPS 连接 来 传输 cookie 


该 函数 设置 Cookie 失败 返回 布尔 值 false， 如 果 设置 Cookie 成 功 ， 则 返回 true。 

由 于 Cookie 与 HITP 协议 的 特定 工作 方式 ， 必 须 在 输出 任何 文本 前 ， 传 送出 所 有 
的 Cookie， 即 在 调用 该 函数 前 ， 程 序 不 能 有 任何 输出 ， 和 否则 PHP 将 会 给 出 警告 信息 ， 
并 且 Cookie 也 不 会 被 传送 。 


如 果 在 同一 个 页 面 中 设置 Cookie, 实际 上 在 设置 Cookie 时 是 按 从 后 往 前 ; 
i 的 顺序 进行 设置 的 ， 如 果 要 先 删除 一 个 Cookie， 再 设置 一 个 Cookie， 则 必须 ! 
， 将 设置 Cookie 的 语句 放 在 前 面 ， 删 除 Cookie 的 语句 放 在 后 面 ， 和 否则 会 出 现 ; 


下 面 示例 演示 了 设置 Cookie 的 值 ， 代 码 如 下 。 


<?php 

echo $s = "我 的 Cookie 值 !"; // 输 出 定义 的 字符 串 
setcookie("testCookie",$s): // 设 置 cookie 
setcookie("testTimeCookie",Ss,time(O+10); /设置 cookie， 失 效 时 间 为 10 秘 
> 


程序 运行 后 会 报错 误 ， 如 图 10-1 所 示 。 


报错 是 因为 在 调用 setcookie 函数 之 前 不 能 有 任何 的 输出 , 所 以 程序 中 应 将 第 2 行 
的 “echo” 去 掉 。 
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Warning: Conmot modify header information - heeders alrcady scnt 
by (cutput started at C:\wamp\mwm\test\inder. php:2) in 
Cr Vwamp wwwNtest\Vindex- php on line 3 


Warning: Cannot modify header information - headers already sent 
by (cutput started at C:\wamp\mm\test\inder. php:2) in 
Cr Wwamp WwwwNtestVindex- php on line 4 
STE i Er 
10-1 


10.2.2 ”访问 Cookie 


PHP 的 $ COOKIE 变量 用 于 取 回 Cookie 的 值 。 
下 面 示例 演示 了 访问 Cookie 的 值 ， 代 码 如 下 。 


<?php 

ifisset($_COOKIE['testCookie])){ 

echo "testCookie 的 值 : ".$_COOKIE['testCookie']."<br>"; 
jelse{ 

echo "testCookie 已 过 期 <br>"; 

上 

ifisset($_COOKIE['testTimeCookie])){ 

echo "testTimeCookie 的 值 : ".$_COOKIE['testTimeCookie']."<br>"; 
jelse{ 

echo "testTimeCookie 已 过 期 <br>"; 

} 

x 


10.2.3 删除 Cookie 


删除 Cookie 可 以 使 用 两 种 方式 : 一 种 是 使 用 一 个 空 值 Cookie 来 实现 ， 即 在 调用 
setcookie 函数 时 不 指定 属性 value 的 值 ; 另 一 种 是 将 过 期 时 间 设 置 为 一 个 过 去 的 时 间 。 
下 面 示例 演示 了 两 种 方法 的 使 用 ， 代 码 如 下 。 


<?php 

setcookie("testCookie"); /第 一 种 方式 : 使 用 setcookie 函数 时 不 指定 属性 value 的 值 
setcookie("testCookie", "", time()-3600);// 第 二 种 方式 : 使 过 期 时 间 为 过 去 的 一 个 时 间 
?> 


10.2.4 ”Cookie 全 局 数组 


前 面 我 们 在 访问 Cookie 的 值 时 使 用 了 一 个 变量 $_ COOKIE, 在 PHP 中 , 提供 了 一 


使 用 PHP 开发 Web 应 用 程序 


个 全 局 数组 $ COOKIE[] 用 于 存储 PHP 的 Cookie 变量 ， 系 统 所 有 的 Cookie 都 保存 在 
这 个 全 局 数组 中 。 设 置 Cookie 就 是 把 所 有 的 Cookie 都 以 键 值 对 的 形式 存 入 该 数组 。 
访问 Cookie 实质 就 是 访问 该 全 局 数组 ， 删 除 指定 的 Cookie 就 是 从 该 数组 中 将 指定 的 
Cookie 值 删除 。 


10.2.5 ”Cookie 综合 案例 


下 面 示例 演示 了 通过 使 用 Cookie 进行 登录 验证 实例 的 分 析 , 其 中 登录 页 面 代 码 如 下 。 


<?php 
iflisset($_COOKIE['logined']) && $ COOKIE['logined]){ 
header("location:welcome.php"); /如 果 已 经 登录 过 ， 则 直接 跳 转 到 欢迎 页 面 
} 
?> 
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0rg/TR/xhtmll/DTD/xhtmll-transitional.dtd"> 
<html xmlns="http://www.w3.org/1999/xhtml"> 
<head> 
<meta http-equiv="Content-Type" content="text/html:; charset=gb2312"> 
<title> 系 统 登 录 </title> 
</head> 
<body> 
<form method="post" action="check.php" name="forml"> <!-- 表单 提交 到 check.php 页 面 
进行 验证 --> 
<table> 
<tr><td> 用 户 名 : </td><td><input type="text" name="username"></td></tr> 
<tr><td> 密 码 : </td><td><input type="password" name="password"></td></tr> 
<tr><td>Cookie 保存 时 间 : </td><td> 
<select name="cookie"> 
<option value="0" checked> 不 保存 </option> 
<option value="1"> 浏 览 器 进程 </option> 
<option value="2"> 一 天 </option> 
<option value="3"> 一 周 </option> 
<option value="4"> 一 月 </option> 
<option value="5"> 一 年 </option> 
</select> 
</td></tr> 
</table> 
<input type="submit" value=" 登 录 " id="submit"/><input type="reset" value=" 重 新 填写 "> 
</form> 
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10-2 
使 用 验证 页 面 check.php 对 用 户 名 和 密码 进行 验证 ， 代 码 如 下 。 


<?php 
$username =$_POST[username']: 
Spassword = $_POST['password']: 
$cookie =$_POST['cookie']; 
// 指 定 一 个 固定 用 户 名 和 密码 admin 和 666666 
if(($username=="admin") && ($password 一 "666666")) { 
switch($cookie){ 
case 1: 
setcookie('logined',1); 
setcookie('username', $username): 
break; 
Case 2: 
setcookie('logined',l],time()+24*60*60); 
setcookie('username', $username,time()+24*60*60); 
break: 
Case 3: 
setcookie('logined',1,time()+24*60*60*7):; 
setcookie('username', $username,time()+24*60*60*7); 
break; 
case 4: 
setcookie('logined', 1,time()+24*60*60*30); 
setcookie('username', $username,time()+24*60*60*30); 
break; 
case 5: 
setcookie('logined', 1,time()+24*60*60*365); 
setcookie('username', $username,time()+24*60*60*365): 
break; 
default: 
setcookie('logined'); 
setcookie('username'); 
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} 
header("location:welcome.php"); /用 户 名 和 密码 正确 则 跳 转 到 欢迎 页 面 
Jelse{ 
echo "用 户 名 或 密码 错误 ， 请 重新 <a hre 人 -login.php'> 登 录 </a>."; /如 果 用 户 名 和 密码 不 正 
确 则 提示 用 户 重 新 登录 


上 
?> 


如 果 用 户 名 或 密码 错误 ， 会 出 现 如 图 10-3 所 示 的 效果 图 。 


用 户 名 或 密码 错 癌 ， 请 重新 登录 . 
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登录 成 功 后 跳 转 到 欢迎 页 面 welcome.php， 页 面 代码 如 下 。 


<?php 
iflisset($_COOKIE['logined']) && $_COOKIE[logined]){ 
echo$ COOKIE["username'] ."， 您 好 ， 欢 迎 光 临 ! "; 
jelse{ 
echo "未 登录 ! 请 <a href='login.php> 登 录 </a>."; 
bp 
> 


页 面 效 果 图 如 图 10-4 所 示 。 
EH 


admin， 您 好 ， 欢 迎 光临 ! 


图 10-4 


上 例 只 是 一 个 非常 简单 的 用 户 登 录 验 证 程序 ， 由 于 将 用 户 的 登录 信息 全 部 存储 在 
Cookie 中 ， 用 户 可 直接 修改 Cookie 文件 ， 伪 造 登录 信息 ， 进 而 达到 通过 验证 的 目的 。 
这 种 将 用 户 登 录 信息 存储 在 客户 端 Cookie 中 的 方式 非常 不 安全 ， 在 实际 的 系统 中 
应 慎 用 。 
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10.3 ” Session 操作 与 应 用 


与 Cookie 一 样 ， 在 PHP 中 同样 可 以 直接 对 Session 进行 操作 ， 如 设置 Session 的 
存储 位 置 、 检 测 变量 是 否 在 Session 注册 、 设 置 Session 的 生命 周期 等 。 


10.3.1 _ Session 的 使 用 


如 果 使 用 了 Session 或 者 在 PHP 文件 中 使 用 Session 变量 , 那么 就 必须 在 调用 之 前 
启动 Session。 启 动 Session 只 需要 在 PHP 脚本 中 使 用 Session_start() 函 数 ，PHP 将 自 
动 完成 Session 文件 的 创建 。 该 函数 语法 如 下 。 


bool session start(void) 


其 中 ，void 表示 函数 没有 参数 。 该 函数 将 开始 一 个 Session 或 者 恢复 已 存在 的 基于 
由 POST、GET 或 者 Cookie 提交 的 当前 SessionID 的 Session。 函 数 总 是 返回 布尔 
值 true。 


| PHP 中 ， 变 量 在 使 用 前 无 须 预先 定义 ， 可 以 直接 在 使 用 时 定义 变量 ， 但 是 在 i 
1 类 中 的 变量 却 需要 先 定义 再 使 用 。 同 时 在 定义 变量 时 ， 也 可 以 不 用 初始 化 变量 ， 
， 未 初始 化 的 变量 具有 其 类 型 的 默认 值 。 


下 面 示例 演示 了 如 何在 PHP 中 启用 Session 并 将 相关 变量 存储 在 Session 中 ， 
代码 如 下 。 
<?php 
session_start(); /开始 Session 
$_SESSION['usemame']="admin"; // 将 一 个 字符 串 存储 在 Session 中 
echo $_ SESSION[username'] 
?> 


运行 结果 如 图 10-5 所 示 。 


tp://127.0.0. 1/test/index. php 
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10.3.2 ”Session 检测 与 注销 


1. Session 检测 
对 于 存储 在 Session 中 的 变量 ， 可 使 用 isset 函数 进行 检测 ， 其 语法 格式 如 下 。 
boolisset(gvar 


其 中 ， 参 数 var 为 要 进行 检测 的 变量 ， 该 函数 如 果 检 测 到 变量 var 存在 则 返回 true， 
否则 返回 false。 
下 面 示 例 演示 了 如 何 使 用 isset 检测 Session， 代 码 如 下 。 


<?php 

session_start(); /开始 Session 

$_SESSION["username']="admin"; // 将 一 个 字符 串 存 储 在 Session 中 
iflisset($_SESSION['username']){ 

echo "用 户 名 已 经 存在 于 Session 中 ， 值 为 :".$_SESSION['username']; 
} 

else{ 

echo "用 户 名 不 存在 1"; 

} 

2 


程序 运行 结果 如 图 10-6 所 示 。 


http://127.0.0. 1/test/index. php 


用 户 名 已 经 存在 于 Session 中 ， 值 为 :admin 


2. Session 注销 
当 需 要 将 已 经 注册 了 的 Session 进行 注销 ， 可 使 用 函数 unset， 其 语法 格式 如 下 。 
void unset($var,.....) 


其 中 ， 参 数 var 为 要 注销 的 Session 名 称 ， 函 数 释 放 给 定 的 变量 ， 即 将 指定 的 Session 
注销 。 
下 面 示例 演示 了 如 何在 PHP 中 使 用 unset 函数 注销 已 有 的 Session， 代 码 如 下 。 
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<?php 

session start(); /开始 Session 

$_SESSION['"username']="admin"; // 将 一 个 字符 串 存 储 在 Session 中 
unset($_SESSION['username"]); 

iflisset($_SESSION["username"]){ 

echo "注销 失败 1"; 

yelse{ 

echo "注销 成 功 !"; 


有 
?> 


运行 结果 如 图 10-7 所 示 。 


3. 注销 所 有 Session 


上 面 只 是 注销 Session 中 的 一 个 名 称 ， 如 果 想 要 注销 整个 Session， 可 使 用 函数 
session_destroy。 其 语法 格式 如 下 。 


bool session_destroy(void) 


该 函数 没有 任何 参数 ， 函 数 将 结束 当前 的 会 话 ， 并 清空 会 话 中 的 所 有 资源 ， 并 且 
总 是 返回 布尔 值 true。 

下 面 示例 演示 了 如 何在 PHP 中 使 用 session_destroy 函数 注销 所 有 的 Session, 代码 
如 下 。 


<?php 
session_ start(); 
$_SESSION['username'] = "username"; 
session_destroy(); 
?> 
上 面 都 是 使 用 PHP 所 提供 的 函数 进行 Session 的 注销 , 还 有 一 种 注销 Session 的 简 
单方 式 ， 就 是 直接 给 超 全 局 变量 $_Session 赋 一 个 空 的 数组 ， 其 语法 格式 如 下 。 


$_SESSION = array0: 
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10.3.3 Session 全 局 数组 


与 Cookie 类 似 ，PHP 提供 了 一 个 超 全 局 数组 用 于 存储 所 有 的 Session 数据 ， 系 统 
中 所 有 的 Session 都 保存 在 这 个 超 全 局 数组 中 。 设 置 Session 就 是 把 所 有 的 Session 都 
以 键 值 对 的 形式 存 入 该 数组 ， 访 问 Session 实质 就 是 访问 该 全 局 数组 ， 注 销 指 定 的 
Session 就 是 从 该 数组 中 将 指定 Session 的 值 删除 。 

下 面 示例 演示 了 Session 全 局 数组 的 使 用 ， 代 码 如 下 。 


<?php 

session_start(); /开始 Session 

$_SESSION[username]="admin"; // 将 一 个 字符 串 存 储 在 Session 中 
$_SESSION['password']="666666"; 

print_r($_SESSION); 


?> 
运行 结果 如 图 10-8 所 示 。 
| EOEAN 


10.3.4 ”Session 综合 案例 


下 面 案例 演示 了 修改 前 面 登录 的 示例 ， 登 录 页 面 login_session.php 代码 如 下 。 


<?php 
session start(); 
iflisset($_SESSION['logined']) && $_SESSION['logined']){ 
header("location:welcome_session.php"); 
} 
全 > 


<!DOCTYPE html PUBLIC "-/W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.org/TR/xhtmll/DTD/xhtml] -transitional.dtd"> 

<html xmlns="http://www.w3.org/1999/xhtml"> 

<head> 

<meta http-equiv="Content-Type" content="text/html; charset=gb2312"> 
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<title> 系 统 登 录 </title> 
</head> 
<body> 
<form method="post" action="check_ session.php" name="form1"> 
<table> 
<tr><td> 用 户 名 : </td><td><input type="text" name="username"></td></tr> 
<tr><td> 密 码 : </td><td><input type="password" name="password"></td></tr> 
</table> 
<input type="submit" value=" 登 录 " id="submit"/><input type="reset" value=" 重 新 填写 "> 
</form> 


</body> 
</html> 


效果 如 图 10-9 所 示 。 


EES3 CD 


验证 页 面 用 于 对 表单 的 数据 进行 验证 ， 如 果 验 证 成 功 ， 则 将 登录 信息 存储 在 Session 
中 ， 然 后 页 面 跳 转 到 欢迎 页 面 ， 如 果 验 证 失败 ， 则 显示 错误 信息 。 
验证 页 面 check_session.php 代码 如 下 。 


<?php 

session_ start(); 

Susername = $_POST['username']: 

Spassword = $_POST['password']; 
if(($username=="admin") && ($password 一 "666666")) { 
$_SESSION['logined'] = true: 

$_SESSION['username'] = $username: 

/lecho "um: $username"; 
header("location:welcome_session.php"); 

Jelse{ 

echo "用 户 名 或 密码 错误 ， 请 重新 <a hre 全 "login_session.php> 登 录 </a>."; 
} 
2 


欢迎 页 面 welcome_session.php 代码 如 下 。 
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<?php 
session start(); 
iflisset($_SESSION['logined']) && $_ SESSION[logined']){ 
echo $_SESSION[usemame] . "， 您 好 ， 欢 迎 光 临 ! <a href-'13.17.php> 退 出 (注销 登 
录 )</a>"; 
jelse{ 
echo "未 登录 ! 请 <a href='login_session.php> 登 录 </a>."; 
b 


?> 


如 果 用 户 登录 成 功 ， 则 显示 如 图 10-10 所 示 的 界面 。 


当 用 户 在 welcome_session.php 欢迎 页 面 中 单 击 “退出 (注销 登录 )” 时 ， 
的 当前 Session 注销 ， 注 销 页 面 destory_session.php 代码 如 下 。 

<2php 

session start(); 

session_destroy(); 


echo "注销 成 功 !<a hre 仁 'login_session.php 人 > 登录 </a>"; 
= 


注销 成 功 后 代码 如 图 10-12 所 示 。 
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10.4 图 像 处 理 


在 实际 应 用 中 , 常常 需要 对 图 片 进行 处 理 , 如 生成 缩 略 图 和 验证 码 图 片 等 .在 PHP 
中 通常 使 用 开源 的 GD 库 来 对 图 片 进行 处 理 。 


10.4.1 图 像 库 简介 


在 PHP 中 ，GD 库 用 于 处 理 图 像 。GD 库 是 一 个 开放 源码 的 动态 创建 图 像 的 函数 
库 。 使 用 GD 库 ， 可 以 创建 和 操作 多 种 不 同 格式 的 图 像 文件 ， 更 为 方便 的 是 ， 可 以 直 


接 以 图 像 流 形 式 将 图 像 输出 到 浏览 器 。 有 关 GD 库 的 详细 信息 大 家 可 以 从 其 官方 网 站 
http://www.libgd.org 了 解 。 


1. 打开 GD 库 


在 PHP 中 要 能 够 使 用 GD 库 进 行 图 像 的 操作 ， 需 要 在 PHP 中 将 GD 库 激 活 。GD 
库 是 PHP 中 默认 进行 安装 的 ， 但 是 没有 被 激活 。 打 开 PHP 的 配置 文件 php.ino， 找 到 
如 下 语句 。 


:extension=php_gd2.dll 


将 前 面 的 分 号 (:) 去 掉 ， 保存 文件 ， 然 后 重启 Apache 服务 器 即 可 。 对 于 该 GD 库 是 
和 否 已 经 成 功 加 载 ， 可 使 用 phpinfo0) 函 数 进行 查看 ,查看 的 GD 库 信息 如 图 10-13 所 示 。 

除了 采用 phpinfo0 显 示 是 否 加 载 GD 库 以 外 ， 还 可 以 使 用 GD 库 自 带 的 gd_info0 
函数 查看 GD 库 信 息 。 


2. GD 库 支 持 的 图 像 格式 


如 果 PHP 服务 器 成 功 安装 了 GD 库 ， 其 版 本 若是 2.0.28 以 上 ， 则 支持 GIF、JPG、 
PNG、WBMP 和 XBM 等 图 像 格 式 ; 若 GD 库 版 本 低 于 1.6， 则 支持 GIF 格式 ， 不 支 
持 PNG 格式 ; 若 GD 库 版 本 在 1.6 至 2.0.28 之 间 ， 则 支持 PNG 格式 ， 不 支持 GIF 格 
式 。 大 家 在 需要 使 用 GD 库 进行 图 像 处 理 时 ， 尽 量 了 解 当前 系统 的 GD 库 版 本 ， 以 做 
相应 的 处 理 。 
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下 面 示例 演示 了 当前 环境 GD 库 是 否 正确 安装 ， 以 及 GD 库 所 支持 的 图 像 文 件 格 
式 ， 代 码 如 下 所 示 。 


<?php 


if (function exists("gd_info")) { 1/ 判断 是 否 安 装 GD 库 
S$gd_info = gd_info(); 


echo "您 的 GD 库 版 本 : ". $gd_info['GD Version'] . "<br>"; 


if ($gd_info['GIF Read Support]) { // 判 断 是 否 支 持 GIF 图 像 读 操作 
echo "支持 GIF 图 像 读 操作 .<br>"; 
}else { 
echo "不 支持 GIF 图 像 读 操作 .<br>"; 
} 
if ($gd_info['GIF Create Support]) { /判断 是 否 支持 GIF 图 像 写 操作 
echo "支持 GIF 图 像 写 操作 .<br>"; 
Jelse { 
echo "不 支持 GIF 图 像 写 操作 .<br>"; 
b 
if ($gd_info['JPG Support]) { // 判 断 是 否 支 持 卫 G 图 像 
echo "支持 JPG 图 像 读 写 操作 .<br>"; 
jelse { 
echo "不 支持 卫 G 图 像 读 写 操作 .<br>"; 
B 
if ($gd_info['PNG Support]) { 1/ 判断 是 否 支持 PNG 图 像 
echo "支持 PNG 图 像 读 写 操作 .<br>"; 
}else { 
echo "不 支持 PNG 图 像 读 写 操作 .<br>"; 
四 
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if (Sgd_info['WBMP Support]) { // 判 断 是 否 支持 WBMP 图 像 
echo "支持 WBMP 图 像 读 写 操作 .<br>"; 

jelse { 
echo "不 支持 WBMP 图 像 读 写 操作 .<br>"; 

} 

if (Sgd_info[' XBM Support]) { 1/ 判断 是 否 支 持 XBM 图 像 
echo "支持 XBM 图 像 读 写 操作 .<br>"; 

jelse { 
echo "不 支持 XBM 图 像 读 写 操作 .<br>"; 

} 

Jelse { 

echo "系统 未 安装 GD 库 ! "; 

} 

ns 


运行 后 结果 如 图 10-14 所 示 。 


10.4.2 ”基本 图 像 处 理 


PHP 中 的 GD 库 提供 了 强大 的 图 像 处 理 功能 。 例 如 ， 将 用 户 上 传 到 服务 器 的 相片 
生成 缩 略 图 以 供 预览 ， 为 防止 恶意 攻击 等 而 使 用 的 验证 码 图 片 ; 给 自己 的 图 片 加 入 版 


权 信 息 水 印 等 。 


1. 创建 图 像 
使 用 GD 库 创建 图 像 ， 首 先 需 要 创建 画布 。 创 建 一 个 真 彩 的 图 像 可 使 用 
imagecreatetruecolor 函数 ， 该 函数 语法 格式 如 下 。 
resource imagecreatetruecolor ( int $x_size , int $y_size ) /创建 一 个 画布 
其 中 ，$x_size 为 所 创建 图 像 的 宽度 ; $y_size 为 所 创建 图 像 的 高 度 。 该 函数 将 返回 一 
个 图 像 标 识 符 ， 其 中 宽 为 Sx_size， 高 为 $y_size。 
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下 面 示例 演示 了 使 用 imagecreatetruecolor 函数 创建 一 个 真 彩 图 像 ， 代 码 如 下 。 


<?php 
header("Content-type: image/png"); 
S$im = @imagecreatetruecolor(200, 100) 
or die("Cannot Initialize new GD image stream"); 
人 


除了 直接 创建 一 个 真 彩 图 像 外 ， 还 可 以 根据 现 有 图 片 文件 创建 图 像 。 从 现 有 的 
PNG 格式 图 像 创 建 图 像 可 使 用 imagecreatefrompng 函数 ， 其 语法 格式 如 下 。 


Tesource imagecreatefrompng ( string $filename ) /根据 现 有 图 片 文 件 创建 图 像 


其 中 , 参数 $filename 为 给 定 的 图 像 名 , 包含 图 像 的 路 径 。 该 函数 返回 一 个 图 像 标识 符 ， 
表示 从 给 定 的 文件 名 取得 的 图 像 。 

在 根据 现 有 图 像 文件 创建 图 像 时 ， 首 先 获取 现 有 图 像 的 文件 类 型 ， 再 根据 现 有 像 
文件 的 类 型 调用 不 同 的 创建 图 像 的 函数 。 


2. 设置 颜色 


在 创建 了 图 像 后 ， 需 要 设置 图 像 的 颜色 ， 设 置 图 像 颜色 使 用 imagecolorallocate 函 
数 ， 其 语法 格式 如 下 。 


JInt imagecolorallocate(resource image,int red,int green,int blue) 


其 中 ， 参 数 image 为 创建 的 图 像 文件 ，red、green 和 blue 分 别 是 所 需要 的 红 、 绿 、 蓝 
三 色 ， 值 从 0 一 255 的 整数 或 十 六 进 制 的 0x00 到 0xFF。 函 数 返 回 一 个 标识 符 ， 代 表 了 
由 给 定 的 RGB 成 分 组 成 的 颜色 。 如 果 颜 色 分 配 失败 ， 则 返回 -1。 


下 面 示例 演示 了 为 已 创建 的 图 像 设置 颜色 ， 代 码 如 下 所 示 。 


<?php 
header("Content-type: image/png"); 
Sim = @imagecreatetruecolor(50, 100) 
or die("Cannot Initialize new GD image stream"); 
Sbgcolor = imagecolorallocate($im, 0, 255. 0): 


单元 十 Cookie、Session 及 图 像 处 理 


S$fontcolor = imagecolorallocate($im, 0, 0, 0); 
imagepng($im); 
?> 


3. 生成 图 像 与 销毁 图 像 


对 于 使 用 imagecreate 系统 函数 创建 的 图 像 ， 还 只 是 一 个 图 像 资 源 ， 不 能 直接 在 济 
览 器 上 显示 。 需 要 使 用 相应 的 函数 将 该 图 像 资 源 输出 到 浏览 器 。 当 需要 使 用 某 种 图 像 
格式 进行 输出 时 ， 就 采用 相应 的 函数 输出 图 像 。 例 如 ， 将 创建 的 图 像 以 PNG 文件 格式 
输出 至 浏览 器 ， 可 使 用 imagepng 函数 ， 其 语法 格式 如 下 。 


bool imagepng ( resource $image [, string $filename ] ) 


imagepng0 将 GD 图 像 流 (image) 以 PNG 格式 输出 到 标准 输出 (通常 为 浏览 器 )， 或 者 如 果 
用 fiename 给 出 了 文件 名 则 将 其 输出 到 该 文件 。 


10.4.3 图像 处 理 案 例 一 一 生成 验证 码 图 片 


在 实际 的 应 用 中 验证 码 使 用 非常 广泛 ， 常 常用 于 网 站 登录 或 评论 。 用 户 在 每 一 次 
提交 表单 时 需要 输入 不 同 的 验证 码 ， 从 而 限制 了 来 自 于 外 部 的 一 些 恶意 操作 。 服 务 器 
在 生成 验证 码 图片 时 ， 将 相应 的 验证 码 保存 在 服务 器 的 Session 中 。 然 后 在 接收 用 户 
所 提交 的 表单 数据 时 ， 判 断 用 户 所 输入 的 验证 码 是 否 与 存放 在 服务 器 Session 中 的 
验证 码 值 相等 。 

下 面 的 random_num.php 示例 演示 了 如 何在 服务 器 端 生 成 验证 码 ， 代 码 如 下 。 


<?php 
header("Content-type:image/png");: // 声 明 图 像 格式 为 png 
session_ start(); 


$authnum = "; 

$str ='abcdefghijkmnpqrstuvwxyz1234567890': // 定 义 随机 的 字母 和 数字 
SstrLength = strlen($str); 

for($i= 1; $i <= 4; $it+) { // 随 机 抽取 4 位 字母 或 数字 
Snum = rand(0, $strLength - 1); 

Sauthnum .= $str[$num]: 

} 


$_SESSION["authnum"] = strtoupper($authnum):; // 将 验证 码 保 存 至 Session 
srand((double)microtime() * 1000000): 
S$im = imagecreate(50, 20): 1/ 创建 图 像 


S$gray = imagecolorallocate($im, 200, 200, 100); /设置 颜色 
$white = imagecolorallocate($im, 255,255,255); /设置 颜色 
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imagefill($im, 10, 5, $egray); 1/ 进行 填充 

$li= imagecolorallocate($im.150, 150, 150); /设置 颜色 

for($i= 0; $i< 3; $i++) { // 加 入 干扰 线 

imageline($im, rand(0, 20), rand(0, 50), rand(20, 40 ), rand(0, 50), $1i); 

} 

for($i=0; $i < strlen($_ SESSION['authnum']); $i++){ ”// 随 机 绘制 验证 码 
$strColor = imagecolorallocate($im,mt _rand(0,100), mt_rand(50,150), 

mt_rand(100,200)); 

SfontSize = mt_rand(3, 5); 

$x = mt_rand(1,5) + 50*8i/4; 

$y = mt rand(1, 5); 

imagestring($im, $fontSize, $x, $y, $_ SESSION['authnunm'][$i], $strColor); 

b 


for($i = 0; $i < 90; $i++) { // 绘 制 干扰 像素 
imagesetpixel($im, rand() % 70, rand() % 30, $gray); 

} 

imagepng($im); // 输 出 PNG 格式 图 片 至 浏览 器 
imagedestroy($im); /销毁 图 像 

?> 


上 面 的 程序 中 ， 产 生 验 证 码 、 绘 制 干扰 线段 、 绘 制 验证 码 、 像 素 等 均 是 采用 随 
机 数 实现 的 ， 这 样 就 能 够 在 页 面 中 展示 多 种 多 样 的 验证 码 图 像 ， 进 而 有 效 地 防止 亚 
意 攻 击 。 

下 面 示例 演示 了 如 何在 页 面 展示 验证 码 ， 并 由 用 户 输入 相应 的 验证 进行 提交 。 
代码 如 下 。 


<?xml version="1.0" encoding="GB18030" ?> 
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.0org/TR/xhtmll/DTD/xhtmll-transitional.dtd"> 
<html xmlns="http://www.w3.org/1999/xhtml"> 
<head> 
<meta http-equiv="Content-Type" content="text/html: charset=GB18030" /> 
<title> 验 证 输入 </title> 
</head> 
<body> 
<form name="forml" method="post" action="check random.php"> 
<table> 
<tr> 
<td> 验 证 码 : <input type="text" name="chkcode" value="" size="15" maxlength="4"></td> 
<td><iframe src="random num.php" height="50px" width="80px" frameborder="0" 
id="chkimg"></iframe> 
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<input type="button" value=" 看 不 清 ， 换 一 张 " onclick="chkimg.location.reload();"> 
</td></tr> 
</table> 
<input type="hidden" value="4" name="checkcnt" /> 
<input type="submit" name="submit1" value=" 提 交 " /> 
</form> 
</body> 
</html> 


运行 后 效果 如 图 10-15 所 示 。 


| 四 htp://ocahost/CookesessonTestrandom haml 


在 上 面 的 程序 中 ， 使 用 这 ame 显示 生成 的 验证 图 片 ， 并 可 以 在 看 不 清楚 时 重新 生 
成 新 的 验证 码 。 显 示 验 证 图 片 的 方式 有 很 多 ， 可 直接 使 用 img 标签 等 。 

下 面 的 程序 check_random.php 演示 了 如 何在 服务 器 接收 表单 提交 的 用 户 输入 的 验 
证 码 并 进行 相关 验证 。 代 码 如 下 : 


<?php 
session_ start(); 


SchkCode = strtoupper($_POST['chkcode']): 
if ($chkCode == $_SESSION['authnum']) { 
echo "验证 成 功 ! "; 

jelse { 

echo "验证 失败 1"; 

} 

session_destroy(); 

Ts 


验证 成 功 后 ， 页 面 效 果 如 图 10-16 所 示 。 


在 上 述 程序 中 ， 需 要 首先 开启 Session， 然 后 获取 表单 提交 的 验证 码 ， 再 判断 用 户 
输入 的 验证 码 是 否 正 确 ， 最 后 再 释放 Session。 
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【单元 小 结 】 


e@ ”Cookie、Session 的 概念 


e@ Cookie、Session 的 操作 与 应 用 
e PHP 中 图 像 处 理 


【单元 自 测 】 


1. 如 何 访问 会 话 变量 (Session)? (  ) 
A. 通过 $_GET B. 通过 $ POST 
C. 通过 $_ REQUEST D. 以 上 都 不 对 

2. 在 忽略 浏览 器 bug 的 正常 情况 下 ， 如 何 用 一 个 与 先前 设置 的 域名 (domain) 不 同 

的 新 域名 来 访问 某 个 Cookie? (  ) 

A. 通过 HTTP_REMOTE_COOKIE 访问 
B. 不 可 能 
C. 在 调用 setcookie0 时 设置 一 个 不 同 的 域名 
D. 向 浏览 器 发 送 额外 的 请 求 

3. index.php 脚本 如 何 访问 表单 元 素 email 的 值 ? ( ””) ( 双 选 ) 
A.$_GET['email] B.$_POST['email] 
C.$_ SESSION['ext] D.$ REQUEST['email] 

. 以 下 脚本 将 如 何 影响 $s 字符 串 ? ( ””) ( 双 选 ) 
<?php 
$s = 'Hello'; 
$ss = htmlentities ($s); 
echo $s; 
ye 


A. 尖 括 号 <> 会 被 转换 成 HTML 标记 ， 因 此 字符 串 将 变 长 
B. 没有 变化 
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C. 在 浏览 器 上 打印 该 字符 串 时 ， 尖 括号 是 可 见 的 
D. 在 浏览 器 上 打印 该 字符 串 时 ， 尖 括号 及 其 内 容 将 被 识别 为 HTML 标签 ， 
此 不 可 见 
5. 如 果 不 给 Cookie 设置 过 期 时 间 会 怎么 样 ? ( ) 
A. 立刻 过 期 B. 永 不 过 期 
C. Cookie 无 法 设置 D. 在 浏览 器 会 话 结束 时 过 期 


【上 机 实战 】 


上 机 目标 


e 掌握 使 用 Session 进行 登录 
e 掌握 采用 JavaScript 与 PHP 进行 交互 


上 机 练习 


人 第 一 阶段 多 


练习 : 使 用 Session 进行 登录 验证 。 


【问题 分 析 】 


编写 页 面 ， 实 现 用 户 名 和 密码 的 验证 ， 验 证 通过 后 将 登录 标志 保存 在 当前 Session 
中 。 其 中 HTML 页面 如 图 10-17 所 示 。 
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代码 如 下 所 示 。 
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<!DOCTYPE html PUBLIC "-WW3CWDTD XHTML 1.0 Transitional//EN" 
"http://www.w3.o0rg/TR/xhtmll/DTD/xhtmll-transitional.dtd"> 
<html xmlns="http://www.w3.org/1999/xhtml"> 
<head> 
<meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> 
<title> 管 理 员 登录 </title> 
<style type="text/css"> 
<!-- 
body,td,th { 
font-size: 16px: 
} 
Sx 


</style></head> 


<body> 
<form id="forml" name="forml" method="post" action="exam 1.php?action=login"> 
<table width="318" height="200" border="0" align="center" cellpadding="0" 
cellspacing="0"> 
<tr> 
<td colspan="2" align="center"> 管 理 员 登 录 </td> 
</tr> 
<tr> 


<td align="left"> 用 户 名 : </td> 
<td align="left"><input name="c_name" type="text" id="c_name" size="30" 
maxlength="20" 
style="width:170px:" /></td> 
</tr> 
<tr> 
<td align="left"> 密 码 : </td> 
<td align="left"><input name="c_pwd" type="password" id="c_pwd" size="30" 
maxlength=" 
style="width:170px:"/></td> 
</tr> 
<tr> 
<td colspan="2" align="center"><input type="submit" name="submit" value=" 提 交 " /> 
<input type="reset" name="submit2" value=" 重 署 " /> 
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验证 的 exam_1.php 页 面 代 码 如 下 。 


<?php 

session start(); 
$c_action=$_GET["action"]; 

S$c_ pwd=trim($ POST["c pwd"]): 


if ($c_action=—"login") 
a ;> pwd=="") 

. echo "<script type='text/javascript'> alert(' 密 码 不 能 为 室 ! ):</script>"; 
} 全 


if ($c_pwd != "phpcoding.cn") 

， echo "<script type='text/javascript> alert(' 密 码 错误 ! );</script>"; 
; exit; 

$_SESSION["admin"]="phpcoding.cn"; 

echo "<script type='text/javascript'> alert( 验 证 通过 !");</script>"; 

< 


令 第 二 阶段 多 


练习 : JavaScript 与 PHP 进行 交互 。 


【问题 分 析 】 

在 实际 应 用 中 ， 常 常 需要 采用 JavaScript 与 PHP 进行 交互 。 例 如 ， 删 除数 据 前 进 
行 一 些 警 告 操作 等 ， 待 用 户 确 认 后 再 进行 实际 的 操作 。 

实现 功能 如 图 10-18 所 示 。 
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代码 如 下 所 示 。 


<head> 

<meta http-equiv="Content-Type" content="text/html: charset=gb2312"> 
<title>PHP 交互 </title> 

<script language="JavaScript"> 

<!-- 

function del chk(message){ 
return confirm(message); 

} 

//--> 

</script> 

</head> 

<body> 

<?php 

S$myrow[guest_name]="guest": 

Smyrow[guest_time]=time(); 

Sguest_name=addslashes($myrow[guest_name]); 

//$guest_name=str2js($Smyrow[guest_name],"™); 

$dele_mess=" 真 的 要 删除 这 个 留言 吗 ? \\n 留言 姓名 : $guest_name($myrow[guest_ip])"."\\n 


echo "delete_ mess=\"$dele mess\""; 

echo "</script>"; 

ye 

<a href="<?php echo "$PHP_SELF?opt=delete"; ?>" onClick='return del chk(delete mess)'> 
删除 留言 

</a> 


</body> 
</html> 
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【拓展 作业 】 


1. 写 一 段 代码 ， 实 现 当 用 户 单 击 不 同 的 链接 注册 不 同 的 Cookie 值 ， 并 根据 不 同 
的 Cookie 值 显示 不 同 的 页 面 样式 时 ， 实 现 页 面 “ 换 肤 ”的 功能 。 

2. 写 一 段 代码 ， 通 过 Session 判断 用 户 是 否 登录 ， 如 果 用 户 没有 登录 则 显示 登录 
界面 供用 户 登录 ; 如果 已 经 登录 则 显示 登录 的 用 户 名 。 
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